нажатие кнопки селена не вызывает встроенного обработчика событий javascript - PullRequest
0 голосов
/ 12 января 2020

My HTML выглядит следующим образом

<div class="container">
    <div class="jumbotron">
        <h1 id="survey-title" class="display-5 text-capitalize">Your favourite candidate</h1>
        <hr class="my-4">

        <div class="card-body">
            <h5 id="question-title" class="card-title">What is your name?</h5>

            <form id="question_form" action="/survey/takesurvey/1/2" method="post" class="form">
                <input type="hidden" name="csrfmiddlewaretoken" value="ELBWYGZRxRuI7CoZ2xkmgCJ3f9JweFMM4Ew1pQbgSE3BLb38VStPJEbHvsyiBEFg">
                <div class="form-group"><input type="text" name="response" value="asdfasdfas" maxlength="400" class="form-control" title="" id="id_response"></div>
            </form>
        </div>

        <a id="btn-previous" href="javascript:history.back()" class="btn btn-primary ">Previous</a>
        <a id="btn-next" href="/survey/takesurvey/finish/1" class="btn btn-primary">Next</a>
    </div>
</div>

Обратите внимание на href на предыдущей кнопке

<a id="btn-previous" href="javascript:history.back()" class="btn btn-primary ">Previous</a>

Нажатие на следующую кнопку отправит форму и go для перехода на следующую страницу, повторное нажатие «предыдущая» должно вернуться и сохранить введенное значение на предыдущей странице. Я использую следующий код для проверки этого поведения. Следующий код работает, только когда я помещаю sleep(1) строку между поиском элемента и щелчком. Иначе страница не возвращается (URL не изменяется)

  • Почему просто ждать изменения URL, прежде чем нажать кнопку, не работает
  • Я подозреваю, что это связано с событием javascript обработчик (встроенный в html) не загружается.
  • Есть ли способ проверить, готова ли страница к событию нажатия
cur_url = self.browser.current_url
self.browser.find_element_by_id('btn-next').click()

wait(lambda: self.assertNotEqual(self.browser.current_url, cur_url))
btn_previous = self.browser.find_element_by_id('btn-previous')

## This code is not working without the sleep below. Which I suspect 
## has to do with javascript event handler (which is inline in html) not being loaded.
## Is there better way to check that the button is ready to be clicked
sleep(1)

btn_previous.click()

Ответы [ 2 ]

1 голос
/ 12 января 2020

Вы можете использовать WebDriverWait с условием element_to_be_clickable. Отметьте здесь также.

from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC

WebDriverWait(self.browser, 5).until(EC.element_to_be_clickable((By.ID, "btn-previous"))).click()

Вы можете попробовать дождаться JavaScript:

WebDriverWait(self.browser, 20).until(lambda d: d.execute_script(
        'return (document.readyState == "complete" || document.readyState == "interactive")'))

# or without interactive 
WebDriverWait(self.browser, 20).until(lambda d: d.execute_script(
        'return document.readyState == "complete"'))
0 голосов
/ 12 января 2020

Это не желаемое решение, а обходной путь, который у меня сейчас работает.

Мои функции ожидания

from time import sleep
from functools import wraps
from selenium.common.exceptions import NoSuchElementException

def wrap_in_wait(func, retires=5):
    @wraps(func)
    def wrapper(*args, **kwargs):
        return wait(func, *args, retires=retires, **kwargs)
    return wrapper

def wait(func, *args, retires=10, exceptions=[AssertionError, NoSuchElementException], **kwargs):
    for i in range(retires):
        try:
            return func(*args, **kwargs)
        except Exception as e:
            print('retry', i)
            if type(e) not in exceptions:
                raise e

            if i == retires-1:
                raise e

            sleep(0.5 + i*.2)


Проверка функции с Обойти решение

def test_question_form_is_loaded_with_existing_answer(self):
        '''Test question form pre-loads'''
        self.browser.get(self.live_server_url + self.url)
        inp = self.browser.find_element_by_css_selector("input#id_response")
        inp.send_keys("This is my answer")

        self.browser.find_element_by_id('btn-next').click()

        @wrap_in_wait
        def click_previous_button():
            btn_previous = self.browser.find_element_by_id('btn-previous')
            btn_previous.click()
            inp = self.browser.find_element_by_css_selector("input#id_response")
            self.assertEqual(inp.get_attribute('value'), "This is my answer")

        click_previous_button()

Это решение повторяет попытку click_previous_button 10 раз, если возникает исключение AssertionError or NoSuchElementException. Он работает со 2-й попытки.

Но я все еще не уверен, почему событие javascript не вызывается, даже когда кнопка активна, и если есть прямой способ проверки, когда javascript готов

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...