2017-03-16 14:27:55 +00:00
|
|
|
##########################################################################
|
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
2021-01-04 10:04:45 +00:00
|
|
|
# Copyright (C) 2013 - 2021, The pgAdmin Development Team
|
2017-03-16 14:27:55 +00:00
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
|
|
|
##########################################################################
|
|
|
|
|
2017-02-22 12:41:28 +00:00
|
|
|
import time
|
2019-08-22 09:20:51 +00:00
|
|
|
import sys
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2019-05-23 08:31:52 +00:00
|
|
|
from selenium import webdriver
|
2017-06-06 11:54:20 +00:00
|
|
|
from selenium.common.exceptions import NoSuchElementException, \
|
2019-03-21 12:04:37 +00:00
|
|
|
WebDriverException, TimeoutException, NoSuchWindowException, \
|
2020-05-21 14:14:28 +00:00
|
|
|
StaleElementReferenceException, ElementNotInteractableException
|
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
|
2019-08-22 09:20:51 +00:00
|
|
|
from regression.feature_utils.locators import QueryToolLocators, \
|
2021-11-10 06:20:20 +00:00
|
|
|
NavMenuLocators, ConnectToServerDiv, PropertyDialogueLocators
|
2019-08-22 09:20:51 +00:00
|
|
|
from regression.feature_utils.tree_area_locators import TreeAreaLocators
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class PgadminPage:
|
|
|
|
"""
|
|
|
|
Helper class for interacting with the page, given a selenium driver
|
|
|
|
"""
|
2020-08-19 07:52:17 +00:00
|
|
|
# Common argument passed for scrolling
|
|
|
|
js_executor_scrollintoview_arg = "arguments[0].scrollIntoView()"
|
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
|
2018-08-21 12:09:36 +00:00
|
|
|
self.app_start_timeout = 90
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# pgAdmin related methods
|
2021-05-27 05:31:25 +00:00
|
|
|
def login_to_app(self, user_detail):
|
|
|
|
if not (self.check_if_element_exist_by_xpath(
|
|
|
|
'//a[@id="navbar-user"]', 1)):
|
|
|
|
user_edt_box_el = self.driver.find_element_by_name('email')
|
|
|
|
user_edt_box_el.send_keys(user_detail['login_username'])
|
|
|
|
password_edt_box_el = self.driver.find_element_by_name('password')
|
|
|
|
password_edt_box_el.send_keys(user_detail['login_password'])
|
|
|
|
submit_btn = self.driver.find_element_by_xpath(
|
|
|
|
'//button[@value="Login"]')
|
|
|
|
submit_btn.click()
|
|
|
|
self.wait_for_spinner_to_disappear()
|
|
|
|
|
2017-03-01 13:20:06 +00:00
|
|
|
def reset_layout(self):
|
2018-08-21 12:09:36 +00:00
|
|
|
attempt = 0
|
|
|
|
while attempt < 4:
|
|
|
|
try:
|
|
|
|
self.click_element(self.find_by_partial_link_text("File"))
|
|
|
|
break
|
|
|
|
except (TimeoutException, NoSuchWindowException):
|
|
|
|
self.driver.refresh()
|
|
|
|
try:
|
|
|
|
WebDriverWait(self.driver, 3).until(EC.alert_is_present())
|
|
|
|
self.driver.switch_to_alert().accept()
|
|
|
|
attempt = attempt + 1
|
|
|
|
except TimeoutException:
|
|
|
|
attempt = attempt + 1
|
|
|
|
|
2017-03-01 13:20:06 +00:00
|
|
|
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
|
|
|
|
2018-10-17 10:50:22 +00:00
|
|
|
def refresh_page(self):
|
2021-05-27 05:31:25 +00:00
|
|
|
try:
|
|
|
|
self.driver.refresh()
|
|
|
|
# wait until alert is present
|
|
|
|
WebDriverWait(self.driver, 1).until(EC.alert_is_present())
|
|
|
|
|
|
|
|
# switch to alert and accept it
|
|
|
|
self.driver.switch_to.alert.accept()
|
|
|
|
except TimeoutException:
|
|
|
|
pass
|
2018-10-17 10:50:22 +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)
|
2018-02-19 11:12:35 +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(
|
2018-02-19 11:12:35 +00:00
|
|
|
"//div[contains(@class, 'alertify') and "
|
|
|
|
"not(contains(@class, 'ajs-hidden'))]//button[.='%s']"
|
2017-06-13 08:50:41 +00:00
|
|
|
% 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):
|
2021-11-10 06:20:20 +00:00
|
|
|
server_group_node = \
|
|
|
|
self.find_by_xpath(TreeAreaLocators.server_group_node("Servers"))
|
|
|
|
ActionChains(self.driver).context_click(server_group_node).perform()
|
|
|
|
ActionChains(self.driver).move_to_element(self.find_by_xpath(
|
|
|
|
TreeAreaLocators.context_menu_element('Create'))).perform()
|
|
|
|
ActionChains(self.driver).move_to_element(self.find_by_xpath(
|
|
|
|
TreeAreaLocators.context_menu_element('Server...'))) \
|
|
|
|
.click().perform()
|
2020-05-21 14:14:28 +00:00
|
|
|
|
|
|
|
WebDriverWait(self.driver, 5).until(EC.visibility_of_element_located(
|
2021-11-10 06:20:20 +00:00
|
|
|
(By.XPATH, PropertyDialogueLocators.server_dialogue_title)))
|
2020-05-21 14:14:28 +00:00
|
|
|
|
|
|
|
# After server dialogue opens
|
2021-11-10 06:20:20 +00:00
|
|
|
self.fill_input_by_field_name(
|
|
|
|
"name", server_config['name'], input_keys=True)
|
|
|
|
self.find_by_xpath(
|
|
|
|
PropertyDialogueLocators.server_connection_tab).click()
|
|
|
|
self.fill_input_by_field_name(
|
|
|
|
"host", server_config['host'], input_keys=True)
|
|
|
|
self.fill_input_by_field_name(
|
|
|
|
"port", server_config['port'], input_keys=True)
|
|
|
|
self.fill_input_by_field_name(
|
|
|
|
"username", server_config['username'], input_keys=True)
|
|
|
|
self.fill_input_by_field_name(
|
|
|
|
"password", server_config['db_password'], input_keys=True)
|
|
|
|
|
|
|
|
save_btn = WebDriverWait(self.driver, 5).until(
|
|
|
|
EC.element_to_be_clickable(
|
|
|
|
(By.XPATH, PropertyDialogueLocators.server_tab_save)))
|
|
|
|
save_btn.click()
|
|
|
|
|
|
|
|
server_tree_xpath = TreeAreaLocators.server_node(server_config['name'])
|
2020-05-11 06:41:31 +00:00
|
|
|
try:
|
|
|
|
WebDriverWait(self.driver, 10).until(
|
|
|
|
EC.visibility_of_element_located(
|
2020-08-19 07:52:17 +00:00
|
|
|
(By.XPATH, server_tree_xpath)))
|
2020-05-11 06:41:31 +00:00
|
|
|
except TimeoutException:
|
2021-11-10 06:20:20 +00:00
|
|
|
self.expand_server_group_node("Server")
|
2020-05-11 06:41:31 +00:00
|
|
|
WebDriverWait(self.driver, 10).until(
|
|
|
|
EC.visibility_of_element_located(
|
2020-08-19 07:52:17 +00:00
|
|
|
(By.XPATH, server_tree_xpath)))
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2017-06-23 13:07:31 +00:00
|
|
|
def open_query_tool(self):
|
2021-11-10 06:20:20 +00:00
|
|
|
self.driver.find_element(By.LINK_TEXT, "Tools").click()
|
|
|
|
tools_menu = self.driver.find_element(By.ID, 'mnu_tools')
|
2017-06-23 13:07:31 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
query_tool = tools_menu.find_element(By.ID, 'query_tool')
|
2017-06-23 13:07:31 +00:00
|
|
|
|
|
|
|
self.enable_menu_item(query_tool, 10)
|
|
|
|
|
|
|
|
self.find_by_partial_link_text("Query Tool").click()
|
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
self.wait_for_element_to_be_visible(
|
|
|
|
self.driver, "//div[@id='btn-conn-status']", 5)
|
|
|
|
|
2019-06-20 11:21:37 +00:00
|
|
|
def open_view_data(self, table_name):
|
|
|
|
self.driver.find_element_by_link_text("Object").click()
|
|
|
|
ActionChains(
|
|
|
|
self.driver
|
|
|
|
).move_to_element(
|
|
|
|
self.driver.find_element_by_link_text("View/Edit Data")
|
|
|
|
).perform()
|
|
|
|
self.find_by_partial_link_text("All Rows").click()
|
|
|
|
time.sleep(1)
|
|
|
|
# wait until datagrid frame is loaded.
|
|
|
|
|
|
|
|
self.click_tab(table_name)
|
|
|
|
|
|
|
|
WebDriverWait(self.driver, self.timeout).until(
|
|
|
|
EC.visibility_of_element_located(
|
|
|
|
(By.CSS_SELECTOR, 'iframe')
|
|
|
|
), "Timed out waiting for div element to appear"
|
|
|
|
)
|
|
|
|
self.driver.switch_to.frame(
|
|
|
|
self.driver.find_element_by_tag_name('iframe')
|
|
|
|
)
|
|
|
|
|
2017-06-23 13:07:31 +00:00
|
|
|
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
|
2018-10-17 10:50:22 +00:00
|
|
|
# two classes 'dropdown-item disabled'.
|
2017-06-23 13:07:31 +00:00
|
|
|
# And if menu is enabled the it will have
|
2018-10-17 10:50:22 +00:00
|
|
|
# only one class 'dropdown-item'.
|
2017-06-23 13:07:31 +00:00
|
|
|
|
2018-10-17 10:50:22 +00:00
|
|
|
if 'dropdown-item' == str(menu_item.get_attribute('class')):
|
2017-06-23 13:07:31 +00:00
|
|
|
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."
|
|
|
|
|
2019-03-21 12:04:37 +00:00
|
|
|
def close_query_tool(self, prompt=True):
|
2017-04-18 12:28:45 +00:00
|
|
|
self.driver.switch_to.default_content()
|
2018-02-19 11:12:35 +00:00
|
|
|
tab = self.find_by_xpath(
|
2019-03-21 12:04:37 +00:00
|
|
|
"//div[@class='wcPanelTab wcPanelTabActive']")
|
2017-04-18 12:28:45 +00:00
|
|
|
ActionChains(self.driver).context_click(tab).perform()
|
2018-02-19 11:12:35 +00:00
|
|
|
self.find_by_xpath(
|
|
|
|
"//li[contains(@class, 'context-menu-item')]/span[contains(text(),"
|
|
|
|
" 'Remove Panel')]").click()
|
2018-03-19 10:58:12 +00:00
|
|
|
if prompt:
|
|
|
|
self.driver.switch_to.frame(
|
2021-11-12 06:26:44 +00:00
|
|
|
self.driver.find_elements(By.TAG_NAME, "iframe")[0])
|
2018-03-19 10:58:12 +00:00
|
|
|
time.sleep(.5)
|
|
|
|
self.click_element(self.find_by_xpath(
|
|
|
|
'//button[contains(@class, "ajs-button") and '
|
|
|
|
'contains(.,"Don\'t save")]'))
|
2020-05-21 14:14:28 +00:00
|
|
|
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
"//button[text()='Rollback']", 1):
|
|
|
|
self.click_element(
|
|
|
|
self.find_by_xpath("//button[text()='Rollback']"))
|
2017-04-18 12:28:45 +00:00
|
|
|
self.driver.switch_to.default_content()
|
|
|
|
|
2019-08-22 09:20:51 +00:00
|
|
|
def clear_query_tool(self):
|
|
|
|
self.click_element(
|
|
|
|
self.find_by_css_selector(QueryToolLocators.btn_clear_dropdown)
|
|
|
|
)
|
|
|
|
ActionChains(self.driver).move_to_element(
|
|
|
|
self.find_by_css_selector(QueryToolLocators.btn_clear)).perform()
|
|
|
|
self.click_element(
|
|
|
|
self.find_by_css_selector(QueryToolLocators.btn_clear)
|
|
|
|
)
|
|
|
|
self.click_modal('Yes')
|
|
|
|
|
|
|
|
def execute_query(self, query):
|
|
|
|
self.fill_codemirror_area_with(query)
|
2019-11-21 11:50:27 +00:00
|
|
|
self.click_execute_query_button()
|
2019-08-22 09:20:51 +00:00
|
|
|
|
2019-12-03 14:05:48 +00:00
|
|
|
def click_execute_query_button(self, timeout=20):
|
2019-11-15 12:32:17 +00:00
|
|
|
retry = 5
|
|
|
|
execute_button = self.find_by_css_selector(
|
|
|
|
QueryToolLocators.btn_execute_query_css)
|
2019-11-20 07:20:04 +00:00
|
|
|
first_click = execute_button.get_attribute('data-click-counter')
|
2019-11-15 12:32:17 +00:00
|
|
|
while retry > 0:
|
|
|
|
execute_button.click()
|
2019-11-20 07:20:04 +00:00
|
|
|
execute_button = self.find_by_css_selector(
|
|
|
|
QueryToolLocators.btn_execute_query_css)
|
|
|
|
second_click = execute_button.get_attribute(
|
|
|
|
'data-click-counter')
|
|
|
|
if first_click != second_click:
|
|
|
|
self.wait_for_query_tool_loading_indicator_to_appear()
|
2019-11-15 12:32:17 +00:00
|
|
|
break
|
|
|
|
else:
|
|
|
|
retry -= 1
|
2019-12-03 14:05:48 +00:00
|
|
|
self.wait_for_query_tool_loading_indicator_to_disappear(timeout)
|
2019-11-15 12:32:17 +00:00
|
|
|
|
2019-08-28 12:34:08 +00:00
|
|
|
def check_execute_option(self, option):
|
|
|
|
""""This function will check auto commit or auto roll back based on
|
|
|
|
user input. If button is already checked, no action will be taken"""
|
2021-11-12 06:26:44 +00:00
|
|
|
query_options = self.driver.find_element(
|
|
|
|
By.CSS_SELECTOR, QueryToolLocators.btn_query_dropdown)
|
2020-05-21 14:14:28 +00:00
|
|
|
expanded = query_options.get_attribute("aria-expanded")
|
|
|
|
if expanded == "false":
|
|
|
|
query_options.click()
|
|
|
|
|
2020-08-19 07:52:17 +00:00
|
|
|
def update_execute_option_setting(
|
|
|
|
css_selector_of_option_status, css_selector_of_option,):
|
|
|
|
retry = 3
|
2021-11-12 06:26:44 +00:00
|
|
|
check_status = self.driver.find_element(
|
|
|
|
By.CSS_SELECTOR, css_selector_of_option_status)
|
2019-08-28 12:34:08 +00:00
|
|
|
if 'visibility-hidden' in check_status.get_attribute('class'):
|
2019-11-20 07:20:04 +00:00
|
|
|
while retry > 0:
|
2020-08-19 07:52:17 +00:00
|
|
|
self.find_by_css_selector(css_selector_of_option).click()
|
2019-11-20 07:20:04 +00:00
|
|
|
time.sleep(0.2)
|
|
|
|
if 'visibility-hidden' not in \
|
|
|
|
check_status.get_attribute('class'):
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
retry -= 1
|
2021-11-10 06:20:20 +00:00
|
|
|
|
2020-08-19 07:52:17 +00:00
|
|
|
if option == 'auto_commit':
|
|
|
|
update_execute_option_setting(
|
|
|
|
QueryToolLocators.btn_auto_commit_check_status,
|
|
|
|
QueryToolLocators.btn_auto_commit)
|
2019-08-28 12:34:08 +00:00
|
|
|
if option == 'auto_rollback':
|
2020-08-19 07:52:17 +00:00
|
|
|
update_execute_option_setting(
|
|
|
|
QueryToolLocators.btn_auto_rollback_check_status,
|
|
|
|
QueryToolLocators.btn_auto_rollback)
|
2019-08-28 12:34:08 +00:00
|
|
|
|
|
|
|
def uncheck_execute_option(self, option):
|
|
|
|
""""This function will uncheck auto commit or auto roll back based on
|
|
|
|
user input. If button is already unchecked, no action will be taken"""
|
2020-05-21 14:14:28 +00:00
|
|
|
query_options = self.driver.find_element_by_css_selector(
|
|
|
|
QueryToolLocators.btn_query_dropdown)
|
|
|
|
expanded = query_options.get_attribute("aria-expanded")
|
|
|
|
if expanded == "false":
|
|
|
|
query_options.click()
|
|
|
|
|
2020-08-19 07:52:17 +00:00
|
|
|
def update_execute_option_setting(
|
|
|
|
css_selector_of_option_status, css_selector_of_option):
|
|
|
|
retry = 3
|
2019-08-28 12:34:08 +00:00
|
|
|
check_status = self.driver.find_element_by_css_selector(
|
2020-08-19 07:52:17 +00:00
|
|
|
css_selector_of_option_status)
|
2019-08-28 12:34:08 +00:00
|
|
|
if 'visibility-hidden' not in check_status.get_attribute('class'):
|
2019-11-20 07:20:04 +00:00
|
|
|
while retry > 0:
|
|
|
|
self.find_by_css_selector(
|
2020-08-19 07:52:17 +00:00
|
|
|
css_selector_of_option).click()
|
2019-11-20 07:20:04 +00:00
|
|
|
time.sleep(0.2)
|
|
|
|
if 'visibility-hidden' in \
|
|
|
|
check_status.get_attribute('class'):
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
retry -= 1
|
2020-08-19 07:52:17 +00:00
|
|
|
|
|
|
|
if option == 'auto_commit':
|
|
|
|
update_execute_option_setting(
|
|
|
|
QueryToolLocators.btn_auto_commit_check_status,
|
|
|
|
QueryToolLocators.btn_auto_commit)
|
2019-08-28 12:34:08 +00:00
|
|
|
if option == 'auto_rollback':
|
2020-08-19 07:52:17 +00:00
|
|
|
update_execute_option_setting(
|
|
|
|
QueryToolLocators.btn_auto_rollback_check_status,
|
|
|
|
QueryToolLocators.btn_auto_rollback)
|
2019-08-28 12:34:08 +00:00
|
|
|
|
2017-06-06 11:54:20 +00:00
|
|
|
def close_data_grid(self):
|
2021-11-10 06:20:20 +00:00
|
|
|
self.driver.switch_to.default_content()
|
2017-06-06 11:54:20 +00:00
|
|
|
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()
|
2021-11-10 06:20:20 +00:00
|
|
|
server_to_remove = self.check_if_element_exists_with_scroll(
|
|
|
|
TreeAreaLocators.server_node(server_config['name']))
|
|
|
|
if server_to_remove:
|
|
|
|
self.driver.execute_script(
|
|
|
|
self.js_executor_scrollintoview_arg, server_to_remove)
|
|
|
|
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("Remove Server")
|
|
|
|
self.click_element(delete_menu_item)
|
|
|
|
self.click_modal('Yes')
|
|
|
|
time.sleep(1)
|
|
|
|
else:
|
|
|
|
print("%s Server is not removed", server_config['name'],
|
|
|
|
file=sys.stderr)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# TODO - Not used to be deleted
|
2017-03-23 11:47:21 +00:00
|
|
|
def select_tree_item(self, tree_item_text):
|
2018-10-17 10:50:22 +00:00
|
|
|
item = self.find_by_xpath(
|
2021-11-10 06:20:20 +00:00
|
|
|
"//*[@id='tree']//*[contains(text(), '" + tree_item_text +
|
|
|
|
"')]/parent::span[@class='aciTreeItem']")
|
2020-08-19 07:52:17 +00:00
|
|
|
self.driver.execute_script(self.js_executor_scrollintoview_arg, item)
|
2019-05-23 08:31:52 +00:00
|
|
|
# unexpected exception like element overlapping, click attempts more
|
|
|
|
# than one time
|
|
|
|
attempts = 3
|
|
|
|
while attempts > 0:
|
|
|
|
try:
|
|
|
|
item.click()
|
|
|
|
break
|
|
|
|
except Exception as e:
|
|
|
|
attempts -= 1
|
|
|
|
time.sleep(.4)
|
|
|
|
if attempts == 0:
|
2020-08-07 07:07:00 +00:00
|
|
|
raise e
|
2017-03-23 11:47:21 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# TODO - Not used to be deleted
|
2019-11-13 05:49:21 +00:00
|
|
|
def click_a_tree_node(self, element_name, list_of_element):
|
|
|
|
"""It will click a tree node eg. server, schema, table name etc
|
|
|
|
will take server name and list of element where this node lies"""
|
|
|
|
operation_status = False
|
|
|
|
elements = list_of_element = self.find_by_xpath_list(
|
|
|
|
list_of_element)
|
|
|
|
if len(elements) > 0:
|
|
|
|
index_of_element = self.get_index_of_element(
|
|
|
|
elements, element_name)
|
|
|
|
if index_of_element >= 0:
|
|
|
|
self.driver.execute_script(
|
2020-08-19 07:52:17 +00:00
|
|
|
self.js_executor_scrollintoview_arg,
|
2019-11-13 05:49:21 +00:00
|
|
|
list_of_element[index_of_element])
|
|
|
|
self.wait_for_elements_to_appear(
|
|
|
|
self.driver, list_of_element[index_of_element])
|
|
|
|
time.sleep(1)
|
2019-11-20 07:20:04 +00:00
|
|
|
list_of_element[index_of_element].click()
|
2019-11-13 05:49:21 +00:00
|
|
|
operation_status = True
|
|
|
|
else:
|
|
|
|
print("{ERROR} - The required element with name: " + str(
|
|
|
|
element_name) +
|
|
|
|
" is not found in function click_a_tree_node, "
|
|
|
|
"so click operation is not performed")
|
|
|
|
else:
|
|
|
|
print("{ERROR} - The element list passed to function "
|
|
|
|
"click_a_tree_node seems empty")
|
|
|
|
return operation_status
|
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
def click_to_expand_tree_node(self, tree_node_web_element,
|
|
|
|
tree_node_exp_check_xpath):
|
|
|
|
"""
|
|
|
|
Method clicks passed webelement to expand specified tree node
|
|
|
|
:param tree_node_web_element:
|
|
|
|
:param tree_node_exp_check_xpath:
|
|
|
|
:return: True is tree_node_exp_check_xpath present in DOM else false
|
|
|
|
"""
|
2021-11-23 12:06:10 +00:00
|
|
|
retry = 2
|
|
|
|
while retry > 0:
|
|
|
|
webdriver.ActionChains(self.driver).double_click(
|
|
|
|
tree_node_web_element).perform()
|
|
|
|
if self.check_if_element_exist_by_xpath(tree_node_exp_check_xpath):
|
|
|
|
return True
|
|
|
|
elif retry == 1:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
time.sleep(1)
|
|
|
|
retry -= 1
|
2021-11-10 06:20:20 +00:00
|
|
|
|
|
|
|
def expand_server_group_node(self, server_group_name):
|
|
|
|
"""
|
|
|
|
Expands specified server group
|
|
|
|
:param server_group_name:
|
|
|
|
:return: True is server group is expanded else false
|
|
|
|
"""
|
|
|
|
server_group_expanded = False
|
2019-03-21 12:04:37 +00:00
|
|
|
self.wait_for_spinner_to_disappear()
|
2021-11-10 06:20:20 +00:00
|
|
|
server_group_node_xpath = \
|
|
|
|
TreeAreaLocators.server_group_node(server_group_name)
|
|
|
|
server_group_node_exp_status_xpath = \
|
|
|
|
TreeAreaLocators.server_group_node_exp_status(server_group_name)
|
|
|
|
server_group_node = self.check_if_element_exists_with_scroll(
|
|
|
|
server_group_node_xpath)
|
|
|
|
|
|
|
|
if server_group_node:
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
server_group_node_exp_status_xpath, 1):
|
|
|
|
server_group_expanded = True
|
2019-08-22 09:20:51 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
server_group_expanded = self.click_to_expand_tree_node(
|
|
|
|
server_group_node, server_group_node_exp_status_xpath)
|
|
|
|
if server_group_expanded:
|
|
|
|
server_group_expanded = True
|
2019-08-22 09:20:51 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("(expand_server_group_node)The Server Group node is "
|
|
|
|
"clicked to expand but it is not expanded",
|
|
|
|
file=sys.stderr)
|
2019-08-22 09:20:51 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("(expand_server_group_node)The Server Group node not found",
|
2019-08-22 09:20:51 +00:00
|
|
|
file=sys.stderr)
|
2021-11-10 06:20:20 +00:00
|
|
|
return server_group_expanded
|
2018-10-17 10:50:22 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
def expand_server_node(self, server_group_name, server_name,
|
|
|
|
server_password):
|
|
|
|
"""
|
|
|
|
Method expands specified server node
|
|
|
|
:param server_group_name: containing server
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
2021-11-18 11:43:32 +00:00
|
|
|
:return: true if server node is expanded else false
|
2021-11-10 06:20:20 +00:00
|
|
|
"""
|
|
|
|
server_expanded = False
|
|
|
|
server_node_xpath = TreeAreaLocators.server_node(server_name)
|
|
|
|
server_node_exp_status_xpath = \
|
|
|
|
TreeAreaLocators.server_node_exp_status(server_name)
|
|
|
|
if self.expand_server_group_node(server_group_name):
|
|
|
|
server_node = \
|
|
|
|
self.check_if_element_exists_with_scroll(server_node_xpath)
|
|
|
|
if server_node:
|
|
|
|
self.driver.execute_script(
|
|
|
|
self.js_executor_scrollintoview_arg, server_node)
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
2021-11-18 11:43:32 +00:00
|
|
|
server_node_exp_status_xpath, 2):
|
2021-11-10 06:20:20 +00:00
|
|
|
server_expanded = True
|
|
|
|
else:
|
|
|
|
server_expanded = self.click_expand_server_node(
|
|
|
|
server_name, server_password, server_node)
|
|
|
|
if not server_expanded:
|
|
|
|
print("(expand_server_node)The Server node is not "
|
2021-11-18 11:43:32 +00:00
|
|
|
"expanded", file=sys.stderr)
|
2021-11-10 06:20:20 +00:00
|
|
|
else:
|
|
|
|
print("(expand_server_node)The Server node not found",
|
|
|
|
file=sys.stderr)
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("The server group node is not expanded", file=sys.stderr)
|
|
|
|
return server_expanded
|
2019-11-11 05:19:00 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
def click_expand_server_node(self, server_name, server_password,
|
|
|
|
server_node):
|
2020-08-19 07:52:17 +00:00
|
|
|
"""
|
|
|
|
Method actually clicks on server node to expand
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
2021-11-10 06:20:20 +00:00
|
|
|
:param server_node:
|
2020-08-19 07:52:17 +00:00
|
|
|
:return: True is click action is successful & server node expanded
|
|
|
|
"""
|
|
|
|
server_node_expansion_status = False
|
2021-11-10 06:20:20 +00:00
|
|
|
if self.check_server_is_connected(server_name):
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
TreeAreaLocators.server_node_exp_status(server_name), 1):
|
2020-08-19 07:52:17 +00:00
|
|
|
server_node_expansion_status = True
|
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
# if server is connected but not expanded
|
2020-08-19 07:52:17 +00:00
|
|
|
webdriver.ActionChains(self.driver).double_click(
|
2021-11-10 06:20:20 +00:00
|
|
|
server_node).perform()
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
2021-11-18 11:43:32 +00:00
|
|
|
TreeAreaLocators.server_node_exp_status(server_name),
|
|
|
|
10):
|
2020-08-19 07:52:17 +00:00
|
|
|
server_node_expansion_status = True
|
2021-11-10 06:20:20 +00:00
|
|
|
else:
|
|
|
|
if self.click_and_connect_server(server_name, server_password):
|
2020-08-19 07:52:17 +00:00
|
|
|
server_node_expansion_status = True
|
2021-11-10 06:20:20 +00:00
|
|
|
else:
|
|
|
|
print("(expand_server_node)The server node is not expanded",
|
|
|
|
file=sys.stderr)
|
2020-08-19 07:52:17 +00:00
|
|
|
return server_node_expansion_status
|
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
def check_server_is_connected(self, server_name):
|
|
|
|
"""
|
|
|
|
This will check connected status of a server"
|
|
|
|
:param server_name:
|
|
|
|
:return: true if server is connected
|
|
|
|
"""
|
|
|
|
server_connected = False
|
|
|
|
try:
|
|
|
|
server_connection_status_element = self.find_by_xpath(
|
|
|
|
TreeAreaLocators.server_connection_status_element(server_name))
|
|
|
|
server_class = server_connection_status_element.get_attribute(
|
|
|
|
'class')
|
|
|
|
if server_class == 'icon-pg' or server_class == 'icon-ppas':
|
|
|
|
server_connected = True
|
|
|
|
except Exception as e:
|
|
|
|
print("There is some exception thrown in the function "
|
|
|
|
"check_server_is_connected and is: " + str(e),
|
2019-11-11 05:19:00 +00:00
|
|
|
file=sys.stderr)
|
2021-11-10 06:20:20 +00:00
|
|
|
return server_connected
|
2019-11-11 05:19:00 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
def click_and_connect_server(self, server_name, password):
|
2020-08-19 07:52:17 +00:00
|
|
|
"""
|
2021-11-10 06:20:20 +00:00
|
|
|
Method will connect to server with password
|
2020-08-19 07:52:17 +00:00
|
|
|
:param server_name:
|
2021-11-10 06:20:20 +00:00
|
|
|
:param password:
|
|
|
|
:return:
|
2020-08-19 07:52:17 +00:00
|
|
|
"""
|
2021-11-10 06:20:20 +00:00
|
|
|
server_connection_status = False
|
|
|
|
try:
|
|
|
|
server_node_ele = self.find_by_xpath(
|
|
|
|
TreeAreaLocators.server_node(server_name))
|
|
|
|
webdriver.ActionChains(self.driver). \
|
|
|
|
double_click(server_node_ele).perform()
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
ConnectToServerDiv.ok_button):
|
|
|
|
field = self.find_by_xpath(
|
|
|
|
ConnectToServerDiv.password_field)
|
|
|
|
self.fill_input(field, password)
|
|
|
|
self.find_by_xpath(ConnectToServerDiv.ok_button).click()
|
|
|
|
self.wait_for_element_to_disappear(
|
|
|
|
lambda driver: driver.find_element_by_xpath(
|
|
|
|
ConnectToServerDiv.ok_button))
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
ConnectToServerDiv.error_message, 2):
|
|
|
|
print(
|
|
|
|
"While entering password in click_and_connect_server "
|
|
|
|
"function, error is occurred : " + str(
|
|
|
|
self.find_by_xpath(
|
|
|
|
ConnectToServerDiv.error_message).text),
|
|
|
|
file=sys.stderr)
|
|
|
|
else:
|
|
|
|
server_connection_status = True
|
|
|
|
except Exception as e:
|
|
|
|
print(
|
|
|
|
"There is some exception thrown click_and_connect_server "
|
|
|
|
"and is: " + str(
|
|
|
|
e), file=sys.stderr)
|
|
|
|
return server_connection_status
|
2020-08-19 07:52:17 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
def expand_databases_node(self, server_group_name, server_name,
|
|
|
|
server_password):
|
2020-08-19 07:52:17 +00:00
|
|
|
"""
|
2021-11-10 06:20:20 +00:00
|
|
|
Method expands Databases node of specfied server
|
|
|
|
:param server_group_name:
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
|
|
|
:return:
|
2020-08-19 07:52:17 +00:00
|
|
|
"""
|
2021-11-10 06:20:20 +00:00
|
|
|
return self.expand_server_child_node(server_group_name, server_name,
|
|
|
|
server_password, "Databases")
|
|
|
|
|
|
|
|
def expand_server_child_node(self, server_group_name, server_name,
|
|
|
|
server_password, server_child_node_name):
|
|
|
|
"""
|
|
|
|
Method expands specified server node's child
|
|
|
|
:param server_group_name:
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
|
|
|
:param server_child_node_name: to be expanded
|
|
|
|
:return: true is child node is expanded else false
|
|
|
|
"""
|
|
|
|
server_child_expanded = False
|
|
|
|
server_child_node_xpath = TreeAreaLocators. \
|
|
|
|
server_child_node(server_name, server_child_node_name)
|
|
|
|
server_child_node_exp_status_xpath = TreeAreaLocators. \
|
|
|
|
server_child_node_exp_status(server_name, server_child_node_name)
|
|
|
|
|
|
|
|
if self.expand_server_node(server_group_name, server_name,
|
|
|
|
server_password):
|
2020-08-19 07:52:17 +00:00
|
|
|
if self.check_if_element_exist_by_xpath(
|
2021-11-10 06:20:20 +00:00
|
|
|
server_child_node_exp_status_xpath, 1):
|
|
|
|
server_child_expanded = True
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
child_node_ele = self.check_if_element_exists_with_scroll(
|
|
|
|
server_child_node_xpath)
|
|
|
|
server_child_expanded = self.click_to_expand_tree_node(
|
|
|
|
child_node_ele, server_child_node_exp_status_xpath)
|
|
|
|
if not server_child_expanded:
|
2021-11-18 11:43:32 +00:00
|
|
|
print("Child not is not expanded after clicking ",
|
2021-11-10 06:20:20 +00:00
|
|
|
file=sys.stderr)
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("The server/previous nodes not expanded", file=sys.stderr)
|
|
|
|
return server_child_expanded
|
|
|
|
|
|
|
|
def expand_database_node(self, server_group_name, server_name,
|
|
|
|
server_password, database_name):
|
|
|
|
"""
|
|
|
|
will expand database node under databases node"
|
|
|
|
:param server_group_name:
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
|
|
|
:param database_name:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
database_expanded = False
|
|
|
|
database_node_xpath = TreeAreaLocators.database_node(database_name)
|
|
|
|
database_node_exp_xpath = \
|
|
|
|
TreeAreaLocators.database_node_exp_status(database_name)
|
|
|
|
|
|
|
|
if self.expand_databases_node(
|
|
|
|
server_group_name, server_name, server_password):
|
|
|
|
database_node = \
|
|
|
|
self.check_if_element_exists_with_scroll(database_node_xpath)
|
|
|
|
if database_node:
|
2019-11-11 05:19:00 +00:00
|
|
|
self.driver.execute_script(
|
2021-11-10 06:20:20 +00:00
|
|
|
self.js_executor_scrollintoview_arg, database_node)
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
database_node_exp_xpath, 1):
|
|
|
|
database_expanded = True
|
|
|
|
else:
|
|
|
|
database_expanded = self.click_to_expand_tree_node(
|
|
|
|
database_node, database_node_exp_xpath)
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("Database node not found - ", file=sys.stderr)
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("The databases/previous nodes not expanded", file=sys.stderr)
|
|
|
|
return database_expanded
|
|
|
|
|
|
|
|
# TODO - We might need this method
|
|
|
|
# def click_to_expand_database_node(self, database_name, database_node):
|
|
|
|
# """
|
|
|
|
# Method clicks on specified database name from expanded databases node
|
|
|
|
# of server.
|
|
|
|
# :param sub_nodes_of_databases_node:
|
|
|
|
# :param index_of_required_db_node:
|
|
|
|
# :param name_of_database:
|
|
|
|
# :return: True if particular database click is successful & expanded
|
|
|
|
# """
|
|
|
|
# database_expanded = False
|
|
|
|
# if self.check_if_element_exist_by_xpath(
|
|
|
|
# TreeAreaLocators.database_node_exp_status(database_name), 2):
|
|
|
|
# database_expanded = True
|
|
|
|
# else:
|
|
|
|
# # TODO - This is bug 6962
|
|
|
|
# webdriver.ActionChains(self.driver).click(database_node).perform()
|
|
|
|
# if self.check_if_element_exist_by_xpath(
|
|
|
|
# TreeAreaLocators.database_node_exp_status(database_name)):
|
|
|
|
# database_expanded = True
|
|
|
|
# print("click_to_expand_database_node> db_node_expanded_status - ",
|
|
|
|
# database_expanded)
|
|
|
|
# return database_expanded
|
|
|
|
#
|
|
|
|
|
|
|
|
def expand_database_child_node(self, server_group_name, server_name,
|
|
|
|
server_password, database_name,
|
|
|
|
database_child_node_name):
|
|
|
|
"""
|
|
|
|
Method expands specified database's child
|
|
|
|
:param server_group_name:
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
|
|
|
:param database_name:
|
|
|
|
:param database_child_node_name:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
database_child_expanded = False
|
|
|
|
database_child_node_xpath = \
|
|
|
|
TreeAreaLocators.database_child_node(
|
|
|
|
database_name, database_child_node_name)
|
|
|
|
database_child_node_exp_status_xpath = \
|
|
|
|
TreeAreaLocators.database_child_node_exp_status(
|
|
|
|
database_name, database_child_node_name)
|
|
|
|
|
|
|
|
if self.expand_database_node(server_group_name, server_name,
|
|
|
|
server_password, database_name):
|
|
|
|
child_node_ele = self.check_if_element_exists_with_scroll(
|
|
|
|
database_child_node_xpath)
|
|
|
|
if child_node_ele:
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
database_child_node_exp_status_xpath, 1):
|
|
|
|
database_child_expanded = True
|
|
|
|
else:
|
|
|
|
database_child_expanded = self.click_to_expand_tree_node(
|
|
|
|
child_node_ele, database_child_node_exp_status_xpath)
|
|
|
|
else:
|
|
|
|
print("Node not found - ", database_child_node_name,
|
|
|
|
file=sys.stderr)
|
|
|
|
else:
|
|
|
|
print("The database/previous nodes not expanded", file=sys.stderr)
|
|
|
|
return database_child_expanded
|
|
|
|
|
|
|
|
def expand_schemas_node(self, server_group_name, server_name,
|
|
|
|
server_password, database_name):
|
|
|
|
"""
|
|
|
|
Method expands Schemas node under specified server & database
|
|
|
|
:param server_group_name:
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
|
|
|
:param database_name:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
return self.expand_database_child_node(server_group_name, server_name,
|
|
|
|
server_password, database_name,
|
|
|
|
"Schemas")
|
|
|
|
|
|
|
|
def expand_schema_node(self, server_group_name, server_name,
|
|
|
|
server_password, database_name, schema_name):
|
|
|
|
"""
|
|
|
|
Method expands schema node
|
|
|
|
:param server_group_name:
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
|
|
|
:param database_name:
|
|
|
|
:param schema_name:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
schema_expanded = False
|
|
|
|
schema_node_xpath = TreeAreaLocators.schema_node(schema_name)
|
|
|
|
schema_node_exp_xpath = TreeAreaLocators.schema_node_exp_status(
|
|
|
|
schema_name)
|
|
|
|
|
|
|
|
if self.expand_schemas_node(server_group_name, server_name,
|
|
|
|
server_password, database_name):
|
|
|
|
schema_node = self.check_if_element_exists_with_scroll(
|
|
|
|
schema_node_xpath)
|
|
|
|
if schema_node:
|
2019-11-11 05:19:00 +00:00
|
|
|
self.driver.execute_script(
|
2021-11-10 06:20:20 +00:00
|
|
|
self.js_executor_scrollintoview_arg, schema_node)
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
schema_node_exp_xpath, 1):
|
|
|
|
schema_expanded = True
|
|
|
|
else:
|
|
|
|
schema_expanded = self.click_to_expand_tree_node(
|
|
|
|
schema_node, schema_node_exp_xpath)
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("Schema node not found - ", file=sys.stderr)
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("The schemas/previous nodes not expanded", file=sys.stderr)
|
|
|
|
return schema_expanded
|
|
|
|
|
|
|
|
def expand_tables_node(self, server_group, server_name, server_password,
|
|
|
|
database_name, schema_name):
|
|
|
|
"""
|
|
|
|
Method expands tables nodes under schema
|
|
|
|
:param server_group:
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
|
|
|
:param database_name:
|
|
|
|
:param schema_name:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
return self.expand_schema_child_node(server_group, server_name,
|
|
|
|
server_password, database_name,
|
|
|
|
schema_name, "Tables")
|
|
|
|
|
|
|
|
def expand_schema_child_node(self, server_group_name, server_name,
|
|
|
|
server_password, database_name, schema_name,
|
|
|
|
schema_child_node_name):
|
|
|
|
"""
|
|
|
|
Expands specified child node of schema
|
|
|
|
:param server_group_name:
|
|
|
|
:param server_name:
|
|
|
|
:param server_password:
|
|
|
|
:param database_name:
|
|
|
|
:param schema_name:
|
|
|
|
:param schema_child_node_name:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
schema_child_expanded = False
|
|
|
|
schema_child_node_xpath = TreeAreaLocators. \
|
|
|
|
schema_child_node(schema_name, schema_child_node_name)
|
|
|
|
schema_child_node_exp_status_check_xpath = TreeAreaLocators. \
|
|
|
|
schema_child_node_exp_status(schema_name, schema_child_node_name)
|
|
|
|
|
|
|
|
if self.expand_schema_node(server_group_name, server_name,
|
|
|
|
server_password, database_name,
|
|
|
|
schema_name):
|
|
|
|
child_node_ele = self.check_if_element_exists_with_scroll(
|
|
|
|
schema_child_node_xpath)
|
|
|
|
if child_node_ele:
|
|
|
|
if self.check_if_element_exist_by_xpath(
|
|
|
|
schema_child_node_exp_status_check_xpath, 1):
|
|
|
|
schema_child_expanded = True
|
|
|
|
else:
|
|
|
|
schema_child_expanded = self.click_to_expand_tree_node(
|
|
|
|
child_node_ele,
|
|
|
|
schema_child_node_exp_status_check_xpath)
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("%s node not found - ", schema_child_node_name,
|
|
|
|
file=sys.stderr)
|
2019-11-11 05:19:00 +00:00
|
|
|
else:
|
2021-11-10 06:20:20 +00:00
|
|
|
print("The schema/previous nodes not expanded", file=sys.stderr)
|
|
|
|
return schema_child_expanded
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# TODO Not used any where to be removed
|
2019-05-23 08:31:52 +00:00
|
|
|
def toggle_open_function_node(self):
|
|
|
|
"""The function will be used for opening Functions node only"""
|
|
|
|
node_expanded = False
|
|
|
|
attempts = 3
|
|
|
|
|
|
|
|
xpath_for_functions_node = \
|
|
|
|
"//span[@class='aciTreeText' and starts-with(text()," \
|
|
|
|
"'Functions')]"
|
|
|
|
xpath_for_exp = "//div[div[div[div[div[div[div[div[span[span[" \
|
|
|
|
"(@class='aciTreeText') and starts-with(text()," \
|
|
|
|
"'Functions')]]]]]]]]]]"
|
|
|
|
xpath_for_button = "//div[span[span[(@class='aciTreeText') " \
|
|
|
|
"and starts-with(text(),'Functions')]]]" \
|
|
|
|
"/span[@class='aciTreeButton']"
|
|
|
|
|
|
|
|
while node_expanded is not True and attempts > 0:
|
|
|
|
# get the element which contains 'aria-expanded' info
|
|
|
|
|
|
|
|
xpath_for_refresh_btn = "//li[@class='context-menu-item']" \
|
2021-10-27 05:35:23 +00:00
|
|
|
"/span[text()='Refresh']"
|
2019-05-23 08:31:52 +00:00
|
|
|
|
2021-10-14 05:43:16 +00:00
|
|
|
# add code to refresh button, sometime the collapsing button
|
2019-05-23 08:31:52 +00:00
|
|
|
# is not visible even if there is sub node.
|
|
|
|
functions_node_ele = self.find_by_xpath(xpath_for_functions_node)
|
|
|
|
|
|
|
|
webdriver.ActionChains(self.driver).move_to_element(
|
|
|
|
functions_node_ele).context_click().perform()
|
|
|
|
refresh_btn = self.find_by_xpath(xpath_for_refresh_btn)
|
|
|
|
refresh_btn.click()
|
|
|
|
time.sleep(.5)
|
|
|
|
|
|
|
|
# get the expansion status
|
|
|
|
function_expansion_ele = self.find_by_xpath(xpath_for_exp)
|
|
|
|
|
|
|
|
# look into the attribute and check if it is already expanded or
|
|
|
|
# not
|
|
|
|
if function_expansion_ele.get_attribute('aria-expanded') \
|
|
|
|
== 'false':
|
|
|
|
# button element of the Function node to open it
|
|
|
|
item_button = self.find_by_xpath(xpath_for_button)
|
|
|
|
ActionChains(self.driver).click(item_button).perform()
|
|
|
|
# Expansion of element on GUI takes sometime, so put small
|
|
|
|
# sleep
|
|
|
|
time.sleep(.5)
|
|
|
|
function_expansion_ele = self.find_by_xpath(
|
|
|
|
xpath_for_exp)
|
|
|
|
if function_expansion_ele.get_attribute('aria-expanded') \
|
|
|
|
== 'true':
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
attempts -= 1
|
|
|
|
else:
|
|
|
|
node_expanded = True
|
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# TODO Not used any where to be removed
|
2019-11-11 05:19:00 +00:00
|
|
|
def get_expansion_status_of_node(self, xpath_node):
|
|
|
|
"""get the expansion status for a node through xpath"""
|
|
|
|
node_is_expanded = False
|
|
|
|
element = self.find_by_xpath(xpath_node)
|
|
|
|
if element.get_attribute("aria-expanded") == 'true':
|
|
|
|
node_is_expanded = True
|
|
|
|
return node_is_expanded
|
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# TODO Not used any where to be removed
|
2019-11-11 05:19:00 +00:00
|
|
|
def get_expansion_status_of_node_element(self, element):
|
|
|
|
"""get the expansion status for an element"""
|
|
|
|
node_is_expanded = False
|
|
|
|
try:
|
|
|
|
if element.get_attribute("aria-expanded") == 'true':
|
|
|
|
node_is_expanded = True
|
|
|
|
except Exception as e:
|
|
|
|
print(
|
|
|
|
"There is some exception thrown in the function "
|
|
|
|
"get_expansion_status_of_node_element and is: " + str(
|
|
|
|
e), file=sys.stderr)
|
|
|
|
return node_is_expanded
|
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# TODO Not used any where to be removed
|
2019-11-11 05:19:00 +00:00
|
|
|
def toggle_open_tree_item(self, tree_item_text):
|
|
|
|
# 'sleep' here helps in cases where underlying nodes are auto opened.
|
|
|
|
# Otherwise, encountered situations where False value is returned
|
|
|
|
# even if the underlying node to be clicked was Opened.
|
|
|
|
time.sleep(.6)
|
|
|
|
item_with_text = self.find_by_xpath(
|
|
|
|
TreeAreaLocators.specified_tree_node.format(tree_item_text))
|
|
|
|
|
2020-08-19 07:52:17 +00:00
|
|
|
self.driver.execute_script(self.js_executor_scrollintoview_arg,
|
2019-11-11 05:19:00 +00:00
|
|
|
item_with_text)
|
|
|
|
|
|
|
|
if item_with_text.find_element_by_xpath(
|
|
|
|
".//ancestor::*[@class='aciTreeLine']").get_attribute(
|
|
|
|
"aria-expanded") == 'false':
|
|
|
|
item = item_with_text.find_element_by_xpath(
|
|
|
|
".//parent::*[@class='aciTreeItem']")
|
|
|
|
ActionChains(self.driver).double_click(item).perform()
|
|
|
|
retry = 3
|
|
|
|
while retry > 0:
|
|
|
|
try:
|
|
|
|
WebDriverWait(self.driver, 5).until((lambda item_with_text: (
|
|
|
|
item_with_text.find_element_by_xpath(
|
|
|
|
".//ancestor::*[@class='aciTreeLine']").
|
|
|
|
get_attribute("aria-expanded") == 'true')))
|
|
|
|
break
|
|
|
|
except TimeoutException:
|
|
|
|
retry -= 1
|
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# TODO Not used any where to be removed
|
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:
|
2018-02-19 11:12:35 +00:00
|
|
|
database_node = driver.find_element_by_xpath(
|
|
|
|
"//*[@id='tree']//*[.='Databases']"
|
|
|
|
"/../*[@class='aciTreeButton']")
|
2017-06-08 12:31:36 +00:00
|
|
|
except WebDriverException:
|
|
|
|
database_node = None
|
|
|
|
|
|
|
|
return dialog is not None or database_node is not None
|
|
|
|
|
|
|
|
self.toggle_open_tree_item(tree_item_text)
|
2018-02-19 11:12:35 +00:00
|
|
|
self._wait_for("Waiting for password dialog or tree to open",
|
|
|
|
check_for_password_dialog_or_tree_open)
|
2017-06-08 12:31:36 +00:00
|
|
|
|
|
|
|
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):
|
2018-02-19 11:12:35 +00:00
|
|
|
return self.wait_for_element(
|
2021-11-10 06:20:20 +00:00
|
|
|
lambda driver: driver.find_element(By.XPATH, xpath)
|
2018-02-19 11:12:35 +00:00
|
|
|
)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
def find_by_id(self, element_id):
|
2018-02-19 11:12:35 +00:00
|
|
|
return self.wait_for_element(
|
2021-11-10 06:20:20 +00:00
|
|
|
lambda driver: driver.find_element(By.ID, element_id)
|
2018-02-19 11:12:35 +00:00
|
|
|
)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2017-06-06 11:54:20 +00:00
|
|
|
def find_by_css_selector(self, css_selector):
|
2018-02-19 11:12:35 +00:00
|
|
|
return self.wait_for_element(
|
2021-11-10 06:20:20 +00:00
|
|
|
lambda driver: driver.find_element(By.CSS_SELECTOR, css_selector)
|
2018-02-19 11:12:35 +00:00
|
|
|
)
|
2017-06-06 11:54:20 +00:00
|
|
|
|
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):
|
2018-02-19 11:12:35 +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
|
|
|
|
|
2018-02-19 11:12:35 +00:00
|
|
|
return self._wait_for(
|
|
|
|
"clicking the element not to throw an exception", click_succeeded
|
|
|
|
)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2019-03-21 12:04:37 +00:00
|
|
|
def js_send_key(self, field, sele_key):
|
|
|
|
keycode = None
|
|
|
|
|
|
|
|
if sele_key in (Keys.RETURN, Keys.ENTER):
|
|
|
|
keycode = 13
|
|
|
|
elif sele_key == Keys.ARROW_DOWN:
|
|
|
|
keycode = 40
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2019-03-21 12:04:37 +00:00
|
|
|
self.driver.execute_script(
|
|
|
|
"arguments[0].dispatchEvent(new KeyboardEvent('keydown', "
|
|
|
|
"{'keyCode':arguments[1], 'which':arguments[1]}));"
|
|
|
|
"arguments[0].dispatchEvent(new KeyboardEvent('keypress', "
|
|
|
|
"{'keyCode':arguments[1], 'which':arguments[1]}));"
|
|
|
|
"arguments[0].dispatchEvent(new KeyboardEvent('keyup', "
|
|
|
|
"{'keyCode':arguments[1], 'which':arguments[1]}));"
|
|
|
|
"arguments[0].dispatchEvent(new Event('input'));"
|
|
|
|
"arguments[0].dispatchEvent(new Event('change'));",
|
|
|
|
field, keycode)
|
|
|
|
|
|
|
|
def js_loose_focus(self, field):
|
|
|
|
self.driver.execute_script(
|
|
|
|
"arguments[0].dispatchEvent(new Event('blur'));", field)
|
|
|
|
|
|
|
|
def fill_input(self, field, field_content, input_keys=False,
|
|
|
|
key_after_input=Keys.ARROW_DOWN):
|
2019-05-23 08:31:52 +00:00
|
|
|
try:
|
|
|
|
attempt = 0
|
|
|
|
for attempt in range(0, 3):
|
|
|
|
field.click()
|
|
|
|
break
|
|
|
|
except Exception as e:
|
|
|
|
time.sleep(.2)
|
|
|
|
if attempt == 2:
|
2020-08-07 07:07:00 +00:00
|
|
|
raise e
|
2019-03-21 12:04:37 +00:00
|
|
|
# Use send keys if input_keys true, else use javascript to set content
|
|
|
|
if input_keys:
|
|
|
|
backspaces = [Keys.BACKSPACE] * len(field.get_attribute('value'))
|
|
|
|
field.send_keys(backspaces)
|
|
|
|
field.send_keys(str(field_content))
|
|
|
|
self.wait_for_input_by_element(field, field_content)
|
|
|
|
else:
|
|
|
|
self.driver.execute_script("arguments[0].value = arguments[1]",
|
|
|
|
field, field_content)
|
|
|
|
# keycode 40 is for arrow down
|
|
|
|
self.js_send_key(field, Keys.ARROW_DOWN)
|
|
|
|
|
|
|
|
if key_after_input:
|
|
|
|
self.js_send_key(field, key_after_input)
|
|
|
|
|
|
|
|
def fill_input_by_field_name(self, field_name, field_content,
|
|
|
|
input_keys=False,
|
|
|
|
key_after_input=Keys.ARROW_DOWN,
|
|
|
|
loose_focus=False):
|
|
|
|
field = self.find_by_css_selector(
|
|
|
|
"input[name='" + field_name + "']:not(:disabled)")
|
|
|
|
self.fill_input(field, field_content, input_keys=input_keys,
|
|
|
|
key_after_input=key_after_input)
|
|
|
|
|
|
|
|
if loose_focus:
|
|
|
|
self.js_loose_focus(field)
|
|
|
|
|
|
|
|
def fill_input_by_css_selector(self, css_selector, field_content,
|
|
|
|
input_keys=False,
|
|
|
|
key_after_input=Keys.ARROW_DOWN,
|
|
|
|
loose_focus=False):
|
|
|
|
field = self.find_by_css_selector(css_selector)
|
|
|
|
self.fill_input(field, field_content, input_keys=input_keys,
|
|
|
|
key_after_input=key_after_input)
|
|
|
|
|
|
|
|
if loose_focus:
|
|
|
|
self.js_loose_focus(field)
|
|
|
|
|
|
|
|
def fill_codemirror_area_with(self, field_content, input_keys=False):
|
2017-06-07 13:26:42 +00:00
|
|
|
def find_codemirror(driver):
|
|
|
|
try:
|
|
|
|
driver.switch_to.default_content()
|
2021-11-10 06:20:20 +00:00
|
|
|
driver.switch_to.frame(
|
2018-02-19 11:12:35 +00:00
|
|
|
driver.find_element_by_tag_name("iframe"))
|
2019-03-21 12:04:37 +00:00
|
|
|
element = driver.find_element_by_css_selector(
|
|
|
|
"#output-panel .CodeMirror")
|
2017-06-07 13:26:42 +00:00
|
|
|
if element.is_displayed() and element.is_enabled():
|
|
|
|
return element
|
|
|
|
except (NoSuchElementException, WebDriverException):
|
|
|
|
return False
|
2021-11-10 06:20:20 +00:00
|
|
|
|
2020-05-11 06:41:31 +00:00
|
|
|
time.sleep(1)
|
2020-05-21 14:14:28 +00:00
|
|
|
self.wait_for_query_tool_loading_indicator_to_disappear(12)
|
2019-03-21 12:04:37 +00:00
|
|
|
|
2020-05-21 14:14:28 +00:00
|
|
|
retry = 2
|
|
|
|
while retry > 0:
|
|
|
|
try:
|
|
|
|
self.driver.switch_to.default_content()
|
|
|
|
WebDriverWait(self.driver, 10).until(
|
|
|
|
EC.frame_to_be_available_and_switch_to_it(
|
|
|
|
(By.TAG_NAME, "iframe")))
|
|
|
|
self.find_by_xpath("//a[text()='Query Editor']").click()
|
|
|
|
|
|
|
|
codemirror_ele = WebDriverWait(
|
|
|
|
self.driver, timeout=self.timeout, poll_frequency=0.01) \
|
|
|
|
.until(find_codemirror,
|
|
|
|
"Timed out waiting for codemirror to appear")
|
|
|
|
codemirror_ele.click()
|
|
|
|
retry = 0
|
|
|
|
except WebDriverException as e:
|
2020-07-20 06:29:28 +00:00
|
|
|
print("Exception in filling code mirror {0} ".format(retry),
|
|
|
|
file=sys.stderr)
|
2020-05-21 14:14:28 +00:00
|
|
|
print(str(e))
|
2020-07-20 06:29:28 +00:00
|
|
|
if retry == 1:
|
2020-05-21 14:14:28 +00:00
|
|
|
raise e
|
|
|
|
retry -= 1
|
2017-06-07 13:26:42 +00:00
|
|
|
|
2019-03-21 12:04:37 +00:00
|
|
|
# Use send keys if input_keys true, else use javascript to set content
|
|
|
|
if input_keys:
|
|
|
|
action = ActionChains(self.driver)
|
|
|
|
action.send_keys(field_content)
|
|
|
|
action.perform()
|
|
|
|
else:
|
|
|
|
self.driver.execute_script(
|
|
|
|
"arguments[0].CodeMirror.setValue(arguments[1]);"
|
|
|
|
"arguments[0].CodeMirror.setCursor("
|
|
|
|
"arguments[0].CodeMirror.lineCount(),0);",
|
|
|
|
codemirror_ele, field_content)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
def click_tab(self, tab_name):
|
2018-06-27 15:40:03 +00:00
|
|
|
WebDriverWait(self.driver, 10).until(EC.element_to_be_clickable(
|
2019-08-22 09:20:51 +00:00
|
|
|
(By.XPATH, NavMenuLocators.select_tab_xpath.format(tab_name))))
|
|
|
|
click_tab = True
|
|
|
|
while click_tab:
|
|
|
|
tab = self.find_by_xpath(
|
|
|
|
NavMenuLocators.select_tab_xpath.format(tab_name))
|
|
|
|
self.click_element(tab)
|
|
|
|
if 'wcPanelTabActive' in tab.get_attribute('class'):
|
|
|
|
break
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2019-03-21 12:04:37 +00:00
|
|
|
def wait_for_input_by_element(self, element, content):
|
|
|
|
def input_field_has_content(driver):
|
|
|
|
return str(content) == element.get_attribute('value')
|
|
|
|
|
|
|
|
return self._wait_for(
|
|
|
|
"field to contain '" + str(content) + "'", input_field_has_content
|
|
|
|
)
|
|
|
|
|
|
|
|
def wait_for_input_field_content(self, field_name, content, wait=1):
|
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')
|
|
|
|
|
2018-02-19 11:12:35 +00:00
|
|
|
return self._wait_for(
|
2019-03-21 12:04:37 +00:00
|
|
|
"field to contain '" + str(content) + "'", input_field_has_content,
|
|
|
|
wait)
|
|
|
|
|
|
|
|
def check_if_element_exist_by_xpath(self, xpath, timeout=5):
|
|
|
|
"""This function will verify if an element exist and on that basis
|
|
|
|
will return True or False. Will handle exception internally"""
|
|
|
|
element_found = False
|
|
|
|
try:
|
|
|
|
WebDriverWait(self.driver, timeout, .01).until(
|
2019-03-21 12:58:52 +00:00
|
|
|
EC.visibility_of_element_located((By.XPATH, xpath)))
|
2019-03-21 12:04:37 +00:00
|
|
|
element_found = True
|
2020-07-24 06:45:29 +00:00
|
|
|
except Exception:
|
2019-03-21 12:04:37 +00:00
|
|
|
pass
|
|
|
|
return element_found
|
2017-02-22 12:41:28 +00:00
|
|
|
|
|
|
|
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)
|
2019-12-04 12:32:53 +00:00
|
|
|
if element.is_displayed():
|
2018-10-17 10:50:22 +00:00
|
|
|
return True
|
2019-03-21 12:04:37 +00:00
|
|
|
except (NoSuchElementException, StaleElementReferenceException):
|
2017-02-22 12:41:28 +00:00
|
|
|
return False
|
|
|
|
|
2018-10-17 10:50:22 +00:00
|
|
|
self._wait_for("element to exist", element_if_it_exists)
|
|
|
|
return find_method_with_args(self.driver)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2019-12-03 14:05:48 +00:00
|
|
|
def wait_for_element_to_disappear(self, find_method_with_args, timeout=5):
|
2017-12-18 15:10:47 +00:00
|
|
|
def element_if_it_disappears(driver):
|
|
|
|
try:
|
|
|
|
element = find_method_with_args(driver)
|
|
|
|
if element.is_displayed() and element.is_enabled():
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
2019-11-15 12:32:17 +00:00
|
|
|
except (NoSuchElementException, StaleElementReferenceException):
|
2017-12-18 15:10:47 +00:00
|
|
|
return True
|
|
|
|
|
2019-12-03 14:05:48 +00:00
|
|
|
return self._wait_for("element to disappear",
|
|
|
|
element_if_it_disappears, timeout)
|
2017-12-18 15:10:47 +00:00
|
|
|
|
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
|
|
|
|
|
2018-02-19 11:12:35 +00:00
|
|
|
self._wait_for("reloading indicator to disappear",
|
|
|
|
reloading_indicator_has_disappeared)
|
2017-03-09 15:34:51 +00:00
|
|
|
|
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:
|
2021-11-10 06:20:20 +00:00
|
|
|
driver.find_element(By.ID, "pg-spinner")
|
2017-02-22 12:41:28 +00:00
|
|
|
return False
|
|
|
|
except NoSuchElementException:
|
|
|
|
return True
|
|
|
|
|
2019-08-22 09:20:51 +00:00
|
|
|
self._wait_for("spinner to disappear", spinner_has_disappeared, 20)
|
2017-02-22 12:41:28 +00:00
|
|
|
|
2019-12-03 14:05:48 +00:00
|
|
|
def wait_for_query_tool_loading_indicator_to_disappear(self, timeout=20):
|
2017-06-27 13:03:04 +00:00
|
|
|
def spinner_has_disappeared(driver):
|
|
|
|
try:
|
2021-06-23 06:52:24 +00:00
|
|
|
# Refer the status message as spinner appears only on the
|
|
|
|
# the data output panel
|
2021-11-10 06:20:20 +00:00
|
|
|
spinner = driver.find_element(
|
|
|
|
By.CSS_SELECTOR,
|
|
|
|
".sql-editor .sql-editor-busy-text-status")
|
2019-03-21 12:04:37 +00:00
|
|
|
return "d-none" in spinner.get_attribute("class")
|
2017-06-27 13:03:04 +00:00
|
|
|
except NoSuchElementException:
|
|
|
|
# wait for loading indicator disappear animation to complete.
|
|
|
|
time.sleep(0.5)
|
|
|
|
return True
|
|
|
|
|
2019-12-03 14:05:48 +00:00
|
|
|
self._wait_for(
|
|
|
|
"spinner to disappear", spinner_has_disappeared, timeout)
|
2019-08-22 09:20:51 +00:00
|
|
|
|
|
|
|
def wait_for_query_tool_loading_indicator_to_appear(self):
|
2019-11-15 12:32:17 +00:00
|
|
|
status = self.check_if_element_exist_by_xpath(
|
2019-08-22 09:20:51 +00:00
|
|
|
"//div[@id='editor-panel']//"
|
2019-11-15 12:32:17 +00:00
|
|
|
"div[@class='pg-sp-container sql-editor-busy-fetching']", 1)
|
|
|
|
return status
|
2017-06-27 13:03:04 +00:00
|
|
|
|
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
|
|
|
|
2018-05-15 14:10:11 +00:00
|
|
|
def wait_for_element_to_reload(self, element_selector):
|
|
|
|
WebDriverWait(self.driver, 20) \
|
|
|
|
.until(EC.staleness_of(element_selector(self.driver)))
|
|
|
|
WebDriverWait(self.driver, 20) \
|
|
|
|
.until_not(EC.staleness_of(element_selector(self.driver)))
|
|
|
|
|
|
|
|
return element_selector(self.driver)
|
|
|
|
|
2018-02-19 11:12:35 +00:00
|
|
|
def _wait_for(self, waiting_for_message, condition_met_function,
|
2019-08-22 09:20:51 +00:00
|
|
|
timeout=5):
|
2017-05-27 18:17:06 +00:00
|
|
|
if timeout is None:
|
|
|
|
timeout = self.timeout
|
2018-02-19 11:12:35 +00:00
|
|
|
return WebDriverWait(self.driver, timeout, 0.01).until(
|
|
|
|
condition_met_function,
|
|
|
|
"Timed out waiting for " + waiting_for_message
|
|
|
|
)
|
2019-08-22 09:20:51 +00:00
|
|
|
|
|
|
|
def wait_for_elements(self, find_method_with_args):
|
|
|
|
"""Using xpath, it will wait for elements"""
|
|
|
|
def element_if_it_exists(driver):
|
|
|
|
try:
|
|
|
|
element = find_method_with_args(driver)
|
2021-11-10 06:20:20 +00:00
|
|
|
if len(element) > 0 and element[0].is_displayed() and \
|
|
|
|
element[0].is_enabled():
|
2019-08-22 09:20:51 +00:00
|
|
|
return element
|
|
|
|
except NoSuchElementException:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return self._wait_for("element to exist", element_if_it_exists)
|
|
|
|
|
2019-11-11 05:19:00 +00:00
|
|
|
def wait_for_elements_to_appear(self, driver, locator, time_value=20):
|
|
|
|
"""This will wait until list of elements or an element is visible,
|
|
|
|
The time out value is userdefined"""
|
|
|
|
elements_located_status = False
|
|
|
|
try:
|
|
|
|
if WebDriverWait(driver, time_value).until(
|
2021-11-10 06:20:20 +00:00
|
|
|
EC.visibility_of_any_elements_located((
|
|
|
|
By.XPATH, locator))):
|
2019-11-11 05:19:00 +00:00
|
|
|
elements_located_status = True
|
2020-07-24 06:45:29 +00:00
|
|
|
except Exception:
|
2019-11-11 05:19:00 +00:00
|
|
|
pass
|
|
|
|
return elements_located_status
|
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
def check_if_element_exists_with_scroll(self, xpath):
|
|
|
|
f_scroll, r_scroll = 111, 111
|
|
|
|
while f_scroll > 0 or r_scroll > 0:
|
|
|
|
try:
|
|
|
|
ele = WebDriverWait(self.driver, 1, 0.01).until(
|
|
|
|
lambda d: d.find_element(By.XPATH, xpath))
|
|
|
|
f_scroll, r_scroll = 0, 0
|
|
|
|
return ele
|
|
|
|
except (TimeoutException, NoSuchElementException) as e:
|
|
|
|
tree_height = int((self.driver.find_element(
|
|
|
|
By.XPATH, "//div[@class='file-tree']/div[1]/div/div").
|
|
|
|
value_of_css_property('height')).split("px")[0])
|
|
|
|
|
|
|
|
if f_scroll == 111 and r_scroll == 111:
|
|
|
|
window_size = int(self.driver.get_window_size()["height"])
|
|
|
|
f_scroll = r_scroll = (tree_height / window_size + 1)
|
|
|
|
|
|
|
|
if f_scroll > 0:
|
|
|
|
bottom_ele = self.driver.find_element(
|
|
|
|
By.XPATH,
|
|
|
|
"//div[@id='tree']/div/div/div/div/div[last()]")
|
|
|
|
bottom_ele_location = int(
|
|
|
|
bottom_ele.value_of_css_property('top').split("px")[0])
|
|
|
|
|
|
|
|
if tree_height - bottom_ele_location < 25:
|
|
|
|
f_scroll = 0
|
|
|
|
else:
|
|
|
|
self.driver.execute_script(
|
|
|
|
self.js_executor_scrollintoview_arg, bottom_ele)
|
|
|
|
f_scroll -= 1
|
|
|
|
elif r_scroll > 0:
|
|
|
|
top_el = self.driver.find_element(
|
|
|
|
By.XPATH,
|
|
|
|
"//div[@id='tree']/div/div/div/div/div[1]")
|
|
|
|
top_el_location = int(
|
|
|
|
top_el.value_of_css_property('top').split("px")[0])
|
|
|
|
|
|
|
|
if (tree_height - top_el_location) == tree_height:
|
|
|
|
r_scroll = 0
|
|
|
|
else:
|
|
|
|
webdriver.ActionChains(self.driver).move_to_element(
|
|
|
|
top_el).perform()
|
|
|
|
r_scroll -= 1
|
|
|
|
else:
|
2021-11-18 11:43:32 +00:00
|
|
|
print("check_if_element_exists_with_scroll > Element NOT found",
|
|
|
|
xpath, file=sys.stderr)
|
2021-11-10 06:20:20 +00:00
|
|
|
return False
|
|
|
|
|
2019-08-22 09:20:51 +00:00
|
|
|
def find_by_xpath_list(self, xpath):
|
|
|
|
"""This will find out list of elements through a single xpath"""
|
|
|
|
return self.wait_for_elements(
|
2021-11-10 06:20:20 +00:00
|
|
|
lambda driver: driver.find_elements(By.XPATH, xpath))
|
2019-08-22 09:20:51 +00:00
|
|
|
|
2021-11-10 06:20:20 +00:00
|
|
|
# TODO Not used any where to be removed
|
2019-08-22 09:20:51 +00:00
|
|
|
def get_index_of_element(self, element_list, target_string):
|
|
|
|
"""it will return index of an element from provided element list"""
|
|
|
|
index_of_required_server = -1
|
|
|
|
if len(element_list) > 0:
|
|
|
|
for index, element in enumerate(element_list):
|
|
|
|
if element.text.startswith(target_string) and (
|
|
|
|
target_string in element.text):
|
|
|
|
index_of_required_server = index
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
print("There seems no record in the provided element list")
|
|
|
|
return index_of_required_server
|
|
|
|
|
|
|
|
def set_switch_box_status(self, switch_box, required_status):
|
|
|
|
"""it will change switch box status to required one. Two elements
|
|
|
|
of the switch boxes are to be provided i) button which is needed to
|
|
|
|
toggle ii) Yes for True or No for False"""
|
|
|
|
status_changed_successfully = False
|
|
|
|
switch_box_element = self.find_by_xpath(switch_box)
|
|
|
|
|
|
|
|
if required_status == 'Yes':
|
2020-08-19 07:52:17 +00:00
|
|
|
status_changed_successfully = \
|
|
|
|
self.toggle_switch_box(switch_box_element,
|
|
|
|
expected_attr_in_class_tag='success',
|
|
|
|
unexpected_attr_in_class_tag='off')
|
2019-08-22 09:20:51 +00:00
|
|
|
else:
|
2020-08-19 07:52:17 +00:00
|
|
|
status_changed_successfully = \
|
|
|
|
self.toggle_switch_box(switch_box_element,
|
|
|
|
expected_attr_in_class_tag='off',
|
|
|
|
unexpected_attr_in_class_tag='success')
|
2019-08-22 09:20:51 +00:00
|
|
|
return status_changed_successfully
|
|
|
|
|
2020-08-19 07:52:17 +00:00
|
|
|
def toggle_switch_box(self, switch_box_ele, expected_attr_in_class_tag,
|
|
|
|
unexpected_attr_in_class_tag):
|
|
|
|
"""
|
|
|
|
Method toggles switch box status using attributes from class tag
|
|
|
|
:param switch_box_ele:
|
|
|
|
:param expected_attr_in_class_tag: e.g. 'off', success
|
|
|
|
:param unexpected_attr_in_class_tag: e.g. 'off', success
|
|
|
|
:return: True if class tag attribute is to expected attribute value
|
|
|
|
in class tag.
|
|
|
|
"""
|
|
|
|
status_changed = False
|
|
|
|
if unexpected_attr_in_class_tag in switch_box_ele.get_attribute(
|
|
|
|
"class"):
|
|
|
|
switch_box_ele.click()
|
|
|
|
time.sleep(1)
|
|
|
|
if expected_attr_in_class_tag in switch_box_ele.get_attribute(
|
|
|
|
"class"):
|
|
|
|
status_changed = True
|
|
|
|
else:
|
|
|
|
print(
|
|
|
|
"(set_switch_box_status)Clicked the "
|
|
|
|
"element to change its status but "
|
|
|
|
"it did not changed",
|
|
|
|
file=sys.stderr)
|
|
|
|
elif expected_attr_in_class_tag in switch_box_ele.get_attribute(
|
|
|
|
"class"):
|
|
|
|
status_changed = True
|
|
|
|
return status_changed
|
|
|
|
|
2019-08-28 12:34:08 +00:00
|
|
|
def retry_click(self, click_locator, verify_locator):
|
2019-08-22 09:20:51 +00:00
|
|
|
click_status = False
|
|
|
|
attempt = 0
|
|
|
|
|
2019-11-11 05:19:00 +00:00
|
|
|
while click_status is not True and attempt < 10:
|
2019-08-22 09:20:51 +00:00
|
|
|
try:
|
2019-08-28 12:34:08 +00:00
|
|
|
element = self.driver.find_element(*click_locator)
|
|
|
|
element.click()
|
2019-12-03 14:05:48 +00:00
|
|
|
WebDriverWait(self.driver, 10).until(
|
2019-08-28 12:34:08 +00:00
|
|
|
EC.visibility_of_element_located(verify_locator))
|
|
|
|
click_status = True
|
2019-08-22 09:20:51 +00:00
|
|
|
except Exception:
|
2019-10-18 06:13:46 +00:00
|
|
|
attempt += 1
|
2019-08-22 09:20:51 +00:00
|
|
|
return click_status
|
2020-05-11 06:41:31 +00:00
|
|
|
|
|
|
|
def paste_values(self, el=None):
|
|
|
|
"""
|
|
|
|
Function paste values in scratch pad
|
|
|
|
:param el:
|
|
|
|
"""
|
|
|
|
actions = ActionChains(self.driver)
|
2020-05-12 11:31:40 +00:00
|
|
|
if "platform" in self.driver.capabilities:
|
|
|
|
platform = (self.driver.capabilities["platform"]).lower()
|
|
|
|
elif "platformName" in self.driver.capabilities:
|
|
|
|
platform = (self.driver.capabilities["platformName"]).lower()
|
2020-05-11 06:41:31 +00:00
|
|
|
if el:
|
|
|
|
# Must step
|
|
|
|
el.click()
|
2020-05-12 11:31:40 +00:00
|
|
|
if 'mac' in platform:
|
2020-05-11 06:41:31 +00:00
|
|
|
# Chrome Step
|
2021-06-23 06:52:24 +00:00
|
|
|
if self.driver.capabilities['browserName'] == 'chrome':
|
|
|
|
actions.key_down(Keys.SHIFT)
|
|
|
|
actions.send_keys(Keys.INSERT)
|
|
|
|
actions.key_up(Keys.SHIFT)
|
|
|
|
actions.perform()
|
|
|
|
else:
|
|
|
|
# FF step
|
|
|
|
el.send_keys(Keys.COMMAND + "v")
|
2020-05-11 06:41:31 +00:00
|
|
|
else:
|
|
|
|
el.send_keys(Keys.CONTROL + "v")
|
|
|
|
|
|
|
|
def wait_for_element_to_be_visible(self, driver, xpath, time_value=20):
|
|
|
|
"""This will wait until an element is visible on page"""
|
|
|
|
element_located_status = False
|
|
|
|
try:
|
|
|
|
if WebDriverWait(driver, time_value).until(
|
|
|
|
EC.visibility_of_element_located((By.XPATH, xpath))):
|
|
|
|
element_located_status = True
|
|
|
|
except TimeoutException:
|
|
|
|
element_located_status = False
|
|
|
|
return element_located_status
|
2021-06-23 06:25:27 +00:00
|
|
|
|
|
|
|
def clear_edit_box(self, edit_box_webelement):
|
|
|
|
while not edit_box_webelement.get_attribute("value") == "":
|
|
|
|
edit_box_webelement.send_keys(Keys.BACK_SPACE)
|