Skip to content

ViewZTPTemplateModalWindow

Модуль modal_view_ztp_template содержит класс для работы с модальным окном шаблона ZTP.

Класс ViewZTPTemplateModalWindow наследует базовый функционал ModalWindowComponent и реализует методы просмотра модального окна шаблона Zero Touch Provisioning.

ViewZTPTemplateModalWindow

Bases: ModalWindowComponent

Модальное окно шаблона Zero Touch Provisioning.

Наследует ModalWindowComponent и добавляет функционал для: 1. Инициализации модального окна с конкретным шаблоном ZTP 2. Закрытия модального окна 3. Получения конфигурационных данных шаблона ZTP 4. Проверки содержимого модального окна 5. Сравнения данных с API специфичными для ZTP

Source code in components_derived\modal_view_ztp_template.py
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
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)

__init__(page, title)

Инициализирует элементы формы модального окна шаблона ZTP.

Source code in components_derived\modal_view_ztp_template.py
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
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")

check_content()

Проверяет наличие и корректность элементов окна ZTP шаблона.

Проверяет: 1. Наличие заголовка окна с именем шаблона 2. Видимость кнопки закрытия 3. Подсказку кнопки закрытия 4. Наличие специфичных полей для ZTP

Source code in components_derived\modal_view_ztp_template.py
53
54
55
56
57
58
59
60
61
62
63
64
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", "Закрыть")

close_window()

Закрывает окно через кнопку 'Закрыть'.

Source code in components_derived\modal_view_ztp_template.py
44
45
46
47
def close_window(self) -> None:
    """Закрывает окно через кнопку 'Закрыть'."""
    close_button = self.get_button_by_name("close")
    close_button.click()

close_window_by_toolbar_button()

Закрывает окно через кнопку в тулбаре.

Source code in components_derived\modal_view_ztp_template.py
49
50
51
def close_window_by_toolbar_button(self):
    """Закрывает окно через кнопку в тулбаре."""
    self.click_toolbar_close_button()

compare_modal_with_api_data(modal_data, api_data, title)

Сравнивает данные из модального окна ZTP с данными из API.

Parameters:

Name Type Description Default
modal_data dict

Данные из модального окна

required
api_data dict

Данные из API ответа

required
title str

Имя шаблона для проверки

required
Source code in components_derived\modal_view_ztp_template.py
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
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)

get_modal_window_data()

Извлекает данные из модального окна шаблона ZTP и структурирует по кодам и значениям.

Returns:

Name Type Description
dict dict

Данные в формате {'код': 'значение'} как в API ZTP

Source code in components_derived\modal_view_ztp_template.py
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
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