Как мне заставить селен последовательно находить и использовать элемент при создании викторины Kahoot?

Я создаю автоматический ответ на викторину Kahoot из электронной таблицы, используя селен. Ниже приведен отрывок из моего кода и отрывок из некоторых df, которые я использую. Все остальное должно работать, однако в строках с 87 по 91 невозможно последовательно создать вопрос-ответ нового типа, а в строках с 98 по 101 невозможно последовательно вводить вопросы и ответы. Если что-то идет не так, то это потому, что веб-элемент не может быть найден, что меня смущает, потому что иногда код работает правильно.

import pandas as pd
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.common.keys import Keys
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC

import openpyxl

# put in spreadsheet
df = pd.read_excel('1000 of the most common Spanish Nouns.xlsx')

# Type in your email or username
email = 'email'
password = 'password'

folder_name = input('Folder Name: ')
quiz_question_length = int(input('Length of each quiz: '))

# set up the webdriver
driver = webdriver.Chrome()

# go to kahoot login
driver.get('https://create.kahoot.it/auth/login')

# type kahoot account credentials
driver.implicitly_wait(10)
driver.find_element(By.ID, 'username').send_keys(f"{email}")
driver.find_element(By.ID, 'password').send_keys(f"{password}")

# get rid of potential popup
try:
    driver.find_element(By.CSS_SELECTOR, 'button[id = "onetrust-pc-btn-handler"]').click()
    driver.find_element(By.CSS_SELECTOR, 'button[class = "ot-pc-refuse-all-handler"]').click()
    print('skipped cookie popup')
except:
    print('no cookie popup')

# press login
driver.find_element(By.CSS_SELECTOR, 'button[id = "login-submit-btn"]').click()

WebDriverWait(driver, 30).until(EC.url_contains("create.kahoot.it"))

# skip second popup
try:
    driver.find_element(By.CSS_SELECTOR, 'button[aria-label = "Next"]').click()
    driver.find_element(By.CSS_SELECTOR, 'button[aria-label = "Next"]').click()
    driver.find_element(By.CSS_SELECTOR, 'a[class = "btn btn--text"]').click()
    print('skipped second popup')
except:
    print('no second popup')

# go to kahoot account's library
driver.implicitly_wait(10)
driver.find_element(By.CSS_SELECTOR, 'a[data-functional-selector = "side-bar-link__library"]').click()

# create a new folder for kahoot quiz
driver.find_element(By.CSS_SELECTOR, 'button[data-functional-selector = "library_side-bar_create-folder"]').click()
driver.find_element(By.CSS_SELECTOR, 'input[placeholder = "New folder"]').send_keys(f'{folder_name}', Keys.ENTER)

# split df into multiple different parts
quizzes = []
for i in range(0, len(df), quiz_question_length):
    if i + quiz_question_length < len(df) + 1:
        quizzes.append(df.iloc[i:i + quiz_question_length])
    else:
        quizzes.append(df.iloc[i:])

# begin creating quizzes
for quiz in quizzes:
    # create new kahoot quiz
    driver.implicitly_wait(10)
    driver.find_element(By.CSS_SELECTOR,
                        'button[class = "button__Button-sc-c6mvr2-0 iwRYhY create-button__PrimaryButton-sc-1wmq7f8-0 jaZrfx"]').click()
    driver.find_element(By.CSS_SELECTOR, 'button[aria-label = "Create new Kahoot."]').click()

    # choose blank option for kahoot quiz
    driver.find_element(By.CSS_SELECTOR, 'button[class = "styles__FlowOptionInnerContainer-sc-hcxgxq-2 gmFgAG"]').click()
    driver.find_element(By.CSS_SELECTOR, 'button[class = "flat-button__FlatButton-sc-6uljam-0 iEWtrJ"]').click()

    # close the theme panel
    driver.find_element(By.CSS_SELECTOR, 'button[aria-label = "Close Themes panel"]').click()

    # PROBLEM WITH ADDING A NEW TYPE ANSWER QUESTION
    for index, row in quiz.iterrows():
        # add new answer type question
        driver.implicitly_wait(5)
        add_question_btn = driver.find_element(By.CSS_SELECTOR, 'button[data-functional-selector = "add-question-button"]')
        add_question_btn.click()
        answer_type_btn = driver.find_element(By.CSS_SELECTOR, 'button[data-functional-selector = "create-button__open-ended"]')
        answer_type_btn.click()

        # set up question and answer input from spreadsheet
        question_input = row['Question']
        answer_input = row['Answer']

        # PROBLEM WITH ADDING QUESTION AND ANSWER TEXT TO QUESTION
        question_text = driver.find_element(By.CSS_SELECTOR, 'p[data-placeholder = "Start typing your question"]')
        question_text.send_keys(question_input)
        answer_text = driver.find_element(By.CSS_SELECTOR, 'input[placeholder = "Type an answer"]')
        answer_text.send_keys(answer_input)
```

df = {
    'Question': ['la explosión', 'el apellido', 'la conferencia', 'la intensidad', 'el paquete'],
    'Answer': ['explosion', 'last name', 'conference', 'intensity', 'package']
}

Я думаю, что мне нужно заставить веб-драйвер подождать, пока он не найдет нужные элементы, которые можно щелкнуть или набрать, потому что элементы определенно есть. Я слышал об использовании явного ожидания для этого, но не знаю, как это сделать. Я новичок в программировании, поэтому это может показаться немного запутанным, и это был всего лишь небольшой проект, над которым я хотел поработать.

ради себя удалите учетные данные для входа из своего кода! А можно поподробнее, какую ошибку вы получаете? какие конкретные строки вызывают исключение, и это NoSuchElementException или что-то еще?

ryyyn 27.06.2024 05:09

извините, я забыл удалить первое упомянутое, а во-вторых, ошибка такая: «selenium.common.Exceptions.StaleElementReferenceException: Сообщение: ссылка на устаревший элемент: устаревший элемент не найден в текущем кадре». Это происходит именно в строках 89, 91, 99 и 101. Здесь происходит либо нажатие кнопки, либо ввод вопроса или ответа.

Richmond Nguyen 27.06.2024 18:30

Здесь либо нажимается кнопка, либо набирается вопрос или ответ, однако я считаю, что ошибка может привести к тому, как я обработал строки, упомянутые выше, но именно здесь я ее и получаю.

Richmond Nguyen 27.06.2024 18:38
Почему в Python есть оператор "pass"?
Почему в Python есть оператор "pass"?
Оператор pass в Python - это простая концепция, которую могут быстро освоить даже новички без опыта программирования.
Некоторые методы, о которых вы не знали, что они существуют в Python
Некоторые методы, о которых вы не знали, что они существуют в Python
Python - самый известный и самый простой в изучении язык в наши дни. Имея широкий спектр применения в области машинного обучения, Data Science,...
Основы Python Часть I
Основы Python Часть I
Вы когда-нибудь задумывались, почему в программах на Python вы видите приведенный ниже код?
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
Алиса и Боб имеют неориентированный граф из n узлов и трех типов ребер:
Оптимизация кода с помощью тернарного оператора Python
Оптимизация кода с помощью тернарного оператора Python
И последнее, что мы хотели бы показать вам, прежде чем двигаться дальше, это
Советы по эффективной веб-разработке с помощью Python
Советы по эффективной веб-разработке с помощью Python
Как веб-разработчик, Python может стать мощным инструментом для создания эффективных и масштабируемых веб-приложений.
1
3
62
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Так что мне удалось решить проблему самостоятельно. Я не уверен, что это самое элегантное решение, но по сути я просто заставляю программу выполнять цикл while, пока упомянутый выше код не заработает, вот он:

    for index, row in quiz.iterrows():
    # add new answer type question
    wait = WebDriverWait(driver, 5)
    while True:
        try:
            add_question_btn = wait.until(EC.element_to_be_clickable((By.CSS_SELECTOR, 'button[data-functional-selector = "add-question-button"]')))
            add_question_btn.click()
            break
        except:
            pass
    while True:
        try:
            answer_type_btn = wait.until(
                EC.element_to_be_clickable(
                    (By.CSS_SELECTOR, 'button[data-functional-selector = "create-button__open-ended"]')))
            answer_type_btn.click()
            break
        except:
            pass

    # set up question and answer input from spreadsheet
    question_input = row['Question']
    answer_input = row['Answer']

    # add questions and answer text to newly created question
    while True:
        try:
            question_text = wait.until(
                EC.element_to_be_clickable((By.CSS_SELECTOR, 'p[data-placeholder = "Start typing your question"]')))
            question_text.send_keys(question_input)
            break
        except:
            pass
    while True:
        try:
            answer_text = wait.until(
                EC.element_to_be_clickable((By.CSS_SELECTOR, 'input[placeholder = "Type an answer"]')))
            answer_text.send_keys(answer_input)
            break
        except:
            pass

Другие вопросы по теме