еренос конкретных файлов: modal_view_template.py, modal_view_ztp_template.py, templates_tab.py, ztp_templates_tab.py, test_ztp_templates_tab.py #1

Merged
RadislavY merged 1 commits from partial-merge-templates into main 2025-10-16 10:42:35 +03:00
5 changed files with 251 additions and 150 deletions

View File

@ -7,8 +7,9 @@
import re
from playwright.sync_api import Page
from tools.logger import get_logger
from locators.modal_window_locators import ModalWindowLocators
from components.modal_window_component import ModalWindowComponent
from components.json_container_component import JsonContainerComponent
from locators.json_container_locators import JsonContainerLocators
logger = get_logger("VIEW_TEMPLATE_MODAL_WINDOW")
@ -19,16 +20,16 @@ class ViewTemplateModalWindow(ModalWindowComponent):
Наследует ModalWindowComponent и добавляет функционал для:
1. Инициализации модального окна с конкретным шаблоном
2. Закрытия модального окна
3. Получения конфигурационных данных шаблона
4. Проверки содержимого модального окна
2. Закрытия модального окна через тулбар
3. Проверки содержимого модального окна
4. Проверки содержимого JSON контейнера
"""
def __init__(self, page: Page, title: str):
"""Инициализирует элементы формы модального окна шаблона."""
super().__init__(page)
# Настройка заголовка и кнопок закрытия
# Настройка заголовка и кнопки закрытия
self.window_title = title
locator_button_toolbar_close = self.page.get_by_role("navigation").filter(
has_text=re.compile(self.window_title)
@ -37,13 +38,8 @@ class ViewTemplateModalWindow(ModalWindowComponent):
self.add_toolbar_title(self.window_title)
self.add_toolbar_button(locator_button_toolbar_close, "close")
locator_button_close = self.page.get_by_role("button", name="Закрыть")
self.add_button(locator_button_close, "close")
def close_window(self) -> None:
"""Закрывает окно через кнопку 'Закрыть'."""
close_button = self.get_button_by_name("close")
close_button.click()
# Инициализация JSON контейнера
self.json_container = JsonContainerComponent(page)
def close_window_by_toolbar_button(self):
"""Закрывает окно через кнопку в тулбаре."""
@ -61,107 +57,18 @@ class ViewTemplateModalWindow(ModalWindowComponent):
self.check_toolbar_button_visibility("close")
self.check_toolbar_button_tooltip("close", "Закрыть")
def get_modal_window_data(self) -> dict:
"""Извлекает данные из модального окна шаблона и структурирует по кодам и значениям.
def verify_json_container_content(self, template_data: dict) -> None:
"""Проверяет соответствие данных контейнера данным из API.
Returns:
dict: Данные в формате {'код': 'значение'} как в API
Args:
template_data: Данные шаблона из API.
"""
modal_data = {}
# Читаем данные из контейнера
actual_data = self.json_container.read_data(JsonContainerLocators.CONTAINER)
# Получаем все значения из input полей
input_locator = self.get_locator(ModalWindowLocators.MODAL_WINDOW_TEXT_FIELD_INPUT)
# Проверка наличия элементов
input_count = input_locator.count()
if input_count == 0:
logger.warning("Поля ввода не найдены в модальном окне")
return modal_data
all_values = []
# Обрабатываем каждое поле с обработкой возможных ошибок
for i in range(input_count):
input_field = input_locator.nth(i)
# Проверяем, что элемент видим и доступен
if not input_field.is_visible():
logger.debug("Поле %s не видимо, пропускаем", i)
continue
# Получаем значение с обработкой возможных ошибок состояния элемента
if input_field.is_visible():
value = input_field.input_value().strip()
if value: # Игнорируем пустые значения
all_values.append(value)
else:
logger.debug("Поле %s стало невидимым после проверки, пропускаем", i)
logger.info("Все значения из полей: %s", all_values)
# Анализируем пары код-значение
i = 0
while i < len(all_values) - 1:
current_value = all_values[i]
next_value = all_values[i + 1]
# Определяем, является ли текущее значение кодом (число)
if current_value.isdigit():
# Текущее значение - код, следующее - значение
modal_data[current_value] = next_value
i += 2 # Перескакиваем через пару
else:
# Если текущее значение не число, ищем следующую пару
i += 1
# Добавляем имя шаблона с ключом 'Шаблон' вместо 'template'
if all_values:
modal_data['Шаблон'] = all_values[-1]
logger.info("Структурированные данные из модального окна: %s", modal_data)
return modal_data
def compare_modal_with_api_data(self, modal_data: dict, api_data: dict,
title: str) -> None:
"""Сравнивает данные из модального окна с данными из API."""
errors = []
# Создаем копию API данных с заменой 'template' на 'Шаблон'
api_data_adapted = api_data.copy()
if 'template' in api_data_adapted:
api_data_adapted['Шаблон'] = api_data_adapted.pop('template')
# Сравниваем все поля
for code, expected_value in api_data_adapted.items():
if code in modal_data:
actual_value = modal_data[code]
if actual_value != expected_value:
error_msg = (
f"Расхождение для кода {code}: "
f"модальное окно='{actual_value}', API='{expected_value}'"
)
logger.error(error_msg)
errors.append(error_msg)
else:
error_msg = f"Код {code} не найден в модальном окне"
logger.error(error_msg)
errors.append(error_msg)
# Дополнительная проверка имени шаблона
modal_template = modal_data.get('Шаблон', '')
if modal_template != title:
error_msg = (
f"Расхождение в имени шаблона: "
f"модальное окно='{modal_template}', ожидается='{title}'"
)
logger.error(error_msg)
errors.append(error_msg)
# Если есть расхождения, выбрасываем ошибку
if errors:
error_details = "\n".join(errors)
assert False, (
f"Обнаружены расхождения для шаблона '{title}':\n{error_details}"
)
logger.info("Данные модального окна соответствуют API для шаблона '%s'", title)
# Сравниваем actual_data с данными конкретного шаблона
self.json_container.check_json_equals(
actual_data,
template_data,
"Expected json content is not equal actual:"
)

View File

@ -0,0 +1,192 @@
"""Модуль modal_view_ztp_template содержит класс для работы с модальным окном шаблона ZTP.
Класс ViewZTPTemplateModalWindow наследует базовый функционал ModalWindowComponent
и реализует методы просмотра модального окна шаблона Zero Touch Provisioning.
"""
import re
from playwright.sync_api import Page
from tools.logger import get_logger
from locators.modal_window_locators import ModalWindowLocators
from components.modal_window_component import ModalWindowComponent
logger = get_logger("VIEW_ZTP_TEMPLATE_MODAL_WINDOW")
class ViewZTPTemplateModalWindow(ModalWindowComponent):
"""Модальное окно шаблона Zero Touch Provisioning.
Наследует ModalWindowComponent и добавляет функционал для:
1. Инициализации модального окна с конкретным шаблоном ZTP
2. Закрытия модального окна
3. Получения конфигурационных данных шаблона ZTP
4. Проверки содержимого модального окна
5. Сравнения данных с API специфичными для ZTP
"""
def __init__(self, page: Page, title: str):
"""Инициализирует элементы формы модального окна шаблона ZTP."""
super().__init__(page)
# Настройка заголовка и кнопок закрытия
self.window_title = title
locator_button_toolbar_close = self.page.get_by_role("navigation").filter(
has_text=re.compile(self.window_title)
).get_by_role("button")
self.add_toolbar_title(self.window_title)
self.add_toolbar_button(locator_button_toolbar_close, "close")
locator_button_close = self.page.get_by_role("button", name="Закрыть")
self.add_button(locator_button_close, "close")
def close_window(self) -> None:
"""Закрывает окно через кнопку 'Закрыть'."""
close_button = self.get_button_by_name("close")
close_button.click()
def close_window_by_toolbar_button(self):
"""Закрывает окно через кнопку в тулбаре."""
self.click_toolbar_close_button()
def check_content(self) -> None:
"""Проверяет наличие и корректность элементов окна ZTP шаблона.
Проверяет:
1. Наличие заголовка окна с именем шаблона
2. Видимость кнопки закрытия
3. Подсказку кнопки закрытия
4. Наличие специфичных полей для ZTP
"""
self.check_by_window_title()
self.check_toolbar_button_visibility("close")
self.check_toolbar_button_tooltip("close", "Закрыть")
def get_modal_window_data(self) -> dict:
"""Извлекает данные из модального окна шаблона ZTP и структурирует по кодам и значениям.
Returns:
dict: Данные в формате {'код': 'значение'} как в API ZTP
"""
modal_data = {}
# Получаем все значения из input полей
input_locator = self.get_locator(ModalWindowLocators.MODAL_WINDOW_TEXT_FIELD_INPUT)
# Проверка наличия элементов
input_count = input_locator.count()
if input_count == 0:
logger.warning("Поля ввода не найдены в модальном окне ZTP")
return modal_data
all_values = []
# Обрабатываем каждое поле с обработкой возможных ошибок
for i in range(input_count):
input_field = input_locator.nth(i)
# Проверяем, что элемент видим и доступен
if not input_field.is_visible():
logger.debug("Поле %s не видимо, пропускаем", i)
continue
# Получаем значение с обработкой возможных ошибок состояния элемента
if input_field.is_visible():
value = input_field.input_value().strip()
if value: # Игнорируем пустые значения
all_values.append(value)
else:
logger.debug("Поле %s стало невидимым после проверки, пропускаем", i)
logger.info("Все значения из полей ZTP шаблона: %s", all_values)
# Анализируем пары код-значение для ZTP формата
i = 0
while i < len(all_values) - 1:
current_value = all_values[i]
next_value = all_values[i + 1]
# Для ZTP шаблонов могут быть как числовые коды, так и строковые идентификаторы
if current_value.isdigit() or self._is_ztp_field_code(current_value):
# Текущее значение - код, следующее - значение
modal_data[current_value] = next_value
i += 2 # Перескакиваем через пару
else:
# Если текущее значение не подходит как код, ищем следующую пару
i += 1
# Добавляем имя шаблона с ключом 'template'
if all_values:
modal_data['template'] = all_values[-1]
logger.info("Структурированные данные из модального окна ZTP: %s", modal_data)
return modal_data
def _is_ztp_field_code(self, value: str) -> bool:
"""Проверяет, является ли значение кодом поля ZTP.
Args:
value: Проверяемое значение
Returns:
bool: True если значение похоже на код поля ZTP
"""
ztp_field_patterns = [
'vendorCode',
'authentication',
'deviceType',
'authenticationOption',
'manufacturer'
]
return any(pattern.lower() in value.lower() for pattern in ztp_field_patterns)
def compare_modal_with_api_data(self, modal_data: dict, api_data: dict,
title: str) -> None:
"""Сравнивает данные из модального окна ZTP с данными из API.
Args:
modal_data: Данные из модального окна
api_data: Данные из API ответа
title: Имя шаблона для проверки
"""
errors = []
# Для ZTP API данные уже содержат нужные ключи
api_data_adapted = api_data.copy()
# Сравниваем все поля
for code, expected_value in api_data_adapted.items():
if code in modal_data:
actual_value = modal_data[code]
if str(actual_value) != str(expected_value):
error_msg = (
f"Расхождение для поля {code}: "
f"модальное окно='{actual_value}', API='{expected_value}'"
)
logger.error(error_msg)
errors.append(error_msg)
else:
error_msg = f"Поле {code} не найдено в модальном окне ZTP"
logger.error(error_msg)
errors.append(error_msg)
# Дополнительная проверка имени шаблона
modal_template = modal_data.get('template', '')
if modal_template != title:
error_msg = (
f"Расхождение в имени шаблона ZTP: "
f"модальное окно='{modal_template}', ожидается='{title}'"
)
logger.error(error_msg)
errors.append(error_msg)
# Если есть расхождения, выбрасываем ошибку
if errors:
error_details = "\n".join(errors)
assert False, (
f"Обнаружены расхождения для ZTP шаблона '{title}':\n{error_details}"
)
logger.info("Данные модального окна ZTP соответствуют API для шаблона '%s'", title)

View File

@ -8,12 +8,10 @@ from playwright.sync_api import Page
from tools.logger import get_logger
from locators.table_locators import TableLocators
from locators.modal_window_locators import ModalWindowLocators
from locators.json_container_locators import JsonContainerLocators
from components_derived.modal_view_template import ViewTemplateModalWindow
from components.modal_window_component import ModalWindowComponent
from components.toolbar_component import ToolbarComponent
from components.table_component import TableComponent
from components.json_container_component import JsonContainerComponent
from pages.base_page import BasePage
logger = get_logger("TEMPLATES_TAB")
@ -39,8 +37,6 @@ class TemplatesTab(BasePage):
self.templates_table = TableComponent(page)
self.modal_windows = {}
self.json_container = JsonContainerComponent(page)
def add_modal_window(self, title: str) -> None:
"""Добавляет модальное окно в коллекцию.
@ -187,6 +183,36 @@ class TemplatesTab(BasePage):
logger.error(error_msg)
assert False, error_msg
def get_template_data_from_api(self, title: str) -> dict:
"""Получает данные шаблона из API.
Args:
title: Имя шаблона.
Returns:
dict: Данные шаблона из API.
"""
# Отправляем запрос к backend для получения информации о шаблоне
response = self.send_get_api_request("e-cmdb/api/device/template")
response_body = self.get_response_body(response)
# Извлекаем конкретный шаблон по имени из ответа API
template_data = self.extract_specific_template(title, response_body)
return template_data
def verify_json_container_content(self, title: str) -> None:
"""Проверяет соответствие данных контейнера данным из API.
Args:
title: Имя шаблона для проверки.
"""
# Получаем данные шаблона из API
template_data = self.get_template_data_from_api(title)
# Получаем модальное окно и проверяем содержимое JSON контейнера
modal_window = self.get_modal_window(title)
modal_window.verify_json_container_content(template_data)
def check_templates_modal_content(self, title: str) -> None:
"""Проверяет наличие и корректность элементов модального окна шаблона.
@ -313,27 +339,3 @@ class TemplatesTab(BasePage):
"""
temp_modal = ModalWindowComponent(self.page)
return temp_modal.check_window_vertical_scrolling()
def verify_json_container_content(self, title: str) -> None:
"""Проверяет соответствие данных контейнера данным из API.
Args:
title: Имя шаблона для проверки.
"""
# Читаем данные из контейнера
actual_data = self.json_container.read_data(JsonContainerLocators.CONTAINER)
# Отправляем запрос к backend для получения информации о шаблоне
response = self.send_get_api_request("e-cmdb/api/device/template")
response_body = self.get_response_body(response)
# Извлекаем конкретный шаблон по имени из ответа API
template_data = self.extract_specific_template(title, response_body)
# Сравниваем actual_data с данными конкретного шаблона
self.json_container.check_json_equals(
actual_data,
template_data,
"Expected json content is not equal actual:"
)

View File

@ -8,7 +8,7 @@ from playwright.sync_api import Page
from tools.logger import get_logger
from locators.table_locators import TableLocators
from locators.modal_window_locators import ModalWindowLocators
from components_derived.modal_view_template import ViewTemplateModalWindow
from components_derived.modal_view_ztp_template import ViewZTPTemplateModalWindow
from components.modal_window_component import ModalWindowComponent
from components.toolbar_component import ToolbarComponent
from components.table_component import TableComponent
@ -44,16 +44,16 @@ class ZTPTemplatesTab(BasePage):
Args:
title: Заголовок окна.
"""
self.modal_windows[title] = ViewTemplateModalWindow(self.page, title)
self.modal_windows[title] = ViewZTPTemplateModalWindow(self.page, title)
def get_modal_window(self, title: str) -> ViewTemplateModalWindow:
def get_modal_window(self, title: str) -> ViewZTPTemplateModalWindow:
"""Возвращает модальное окно по заголовку.
Args:
title: Заголовок окна.
Returns:
ViewTemplateModalWindow: Экземпляр модального окна шаблона.
ViewZTPTemplateModalWindow: Экземпляр модального окна шаблона.
Raises:
AssertionError: Если окно не найдено.
@ -92,7 +92,7 @@ class ZTPTemplatesTab(BasePage):
row_locator.click()
# Создаем временный экземпляр модального окна для получения заголовка
temp_modal = ViewTemplateModalWindow(self.page, "")
temp_modal = ViewZTPTemplateModalWindow(self.page, "")
title = temp_modal.toolbar.get_toolbar_title_text(
ModalWindowLocators.MODAL_WINDOW_TITLE
)
@ -150,7 +150,7 @@ class ZTPTemplatesTab(BasePage):
temp_modal = ModalWindowComponent(self.page)
temp_modal.scroll_window_down()
def check_templates_modal_content(self, title: str) -> None:
def check_ztp_templates_modal_content(self, title: str) -> None:
"""Проверяет наличие и корректность элементов модального окна шаблона.
Args:
@ -162,7 +162,7 @@ class ZTPTemplatesTab(BasePage):
modal_window = self.get_modal_window(title)
modal_window.check_content()
def check_templates_table_content(self) -> None:
def check_ztp_templates_table_content(self) -> None:
"""Проверяет содержимое таблицы шаблонов.
Проверяет заголовки и наличие данных в таблице.

View File

@ -75,7 +75,7 @@ class TestZTPTemplatesTab:
browser.wait_for_timeout(5000)
# Проверка содержимого таблицы шаблонов
ztp_templates_tab.check_templates_table_content()
ztp_templates_tab.check_ztp_templates_table_content()
#@pytest.mark.skip(reason=" Временно исключено из тестирования")
def test_templates_table_row_highlighting(self, browser: Page) -> None:
@ -226,7 +226,7 @@ class TestZTPTemplatesTab:
ztp_templates_tab.should_be_modal_window()
# Проверка содержимого модального окна
ztp_templates_tab.check_templates_modal_content(title)
ztp_templates_tab.check_ztp_templates_modal_content(title)
# Закрытие модального окна через кнопку 'Закрыть'
ztp_templates_tab.close_modal_window(title)