Я создаю автоматический ответ на викторину 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']
}
Я думаю, что мне нужно заставить веб-драйвер подождать, пока он не найдет нужные элементы, которые можно щелкнуть или набрать, потому что элементы определенно есть. Я слышал об использовании явного ожидания для этого, но не знаю, как это сделать. Я новичок в программировании, поэтому это может показаться немного запутанным, и это был всего лишь небольшой проект, над которым я хотел поработать.
извините, я забыл удалить первое упомянутое, а во-вторых, ошибка такая: «selenium.common.Exceptions.StaleElementReferenceException: Сообщение: ссылка на устаревший элемент: устаревший элемент не найден в текущем кадре». Это происходит именно в строках 89, 91, 99 и 101. Здесь происходит либо нажатие кнопки, либо ввод вопроса или ответа.
Здесь либо нажимается кнопка, либо набирается вопрос или ответ, однако я считаю, что ошибка может привести к тому, как я обработал строки, упомянутые выше, но именно здесь я ее и получаю.
Так что мне удалось решить проблему самостоятельно. Я не уверен, что это самое элегантное решение, но по сути я просто заставляю программу выполнять цикл 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
ради себя удалите учетные данные для входа из своего кода! А можно поподробнее, какую ошибку вы получаете? какие конкретные строки вызывают исключение, и это
NoSuchElementException
или что-то еще?