This repository has been archived by the owner on Jun 6, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 40
/
browsers.py
146 lines (134 loc) · 6.41 KB
/
browsers.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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
import logging
import time
from asyncio import TimeoutError
from selenium import webdriver
from selenium.common.exceptions import TimeoutException, NoSuchElementException
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By
from selenium.webdriver.common.keys import Keys
from selenium.webdriver.firefox.options import Options
from selenium.webdriver.support.wait import WebDriverWait
import config
class Browser:
def __init__(self):
"""
Represents a Selenium browser object with added functions for Discord-specific purposes.
No parameters; global settings are used from config.py.
Methods
-----------
browser_login()
Opens the browser to the Discord channel and logs in if necessary.
Returns True if load successful.
Raises TimeoutError if the page does not load.
Raises ValueError if incorrect page loads.
send_text(text)
Send text to active channel.
react_emoji(emoji, message_id)
Searches and clicks an emoji button.
Raises Exception if button was not found.
roll(count)
Sends the roll command for count number of times with 3 seconds between each roll.
refresh()
Refreshes the page.
close()
Closes the browser window.
"""
# Selenium browser control here
options = Options()
options.add_argument('-headless')
self.driver = webdriver.Firefox(executable_path=config.WEB_DRIVER_PATH, options=options)
self.actions = ActionChains(self.driver)
# Logging initialization
self.logger = logging.getLogger(__name__)
# Initiate browser
def browser_login(self):
self.logger.info('Browser thread started')
self.logger.info('Attempting to open Discord in browser')
self.driver.get(f'https://discord.com/channels/{config.SERVER_ID}/{config.CHANNEL_ID}')
try:
email = WebDriverWait(self.driver, 10).until(lambda x: x.find_element(By.NAME, 'email'))
except TimeoutException:
if f'{config.SERVER_ID}/{config.CHANNEL_ID}' not in self.driver.current_url:
# No login screen, but wrong channel (some weird error)
self.logger.critical('The channel did not load and no login was asked!')
raise TimeoutError
else:
self.logger.info('Logging in with provided credentials (this may take up to 30 seconds)')
email.send_keys(config.LOGIN_INFO[0])
self.driver.find_element(By.NAME, 'password').send_keys(config.LOGIN_INFO[1])
self.driver.find_element(By.XPATH, "//button[@type='submit']").click()
try:
# Wait for main screen
WebDriverWait(self.driver, 30).until(lambda x: x.find_element(By.CLASS_NAME, 'slateTextArea-1Mkdgw'))
if f'{config.SERVER_ID}/{config.CHANNEL_ID}' not in self.driver.current_url:
# Logged in, but wrong channel (some weird error)
raise ValueError
except TimeoutException or NoSuchElementException:
self.logger.critical('Login was unsuccessful. Please check LOGIN_INFO entry in config.py')
raise TimeoutError
except ValueError:
self.logger.critical('Login was successful but the channel did not load')
raise ValueError
else:
self.logger.info(f'Login successful to server {config.SERVER_ID} and channel {config.CHANNEL_ID}')
return True
def send_text(self, text: str):
# For some reason, typing directly into the message box doesn't work
# ActionChains must be used instead to type character by character
self.actions = ActionChains(self.driver)
self.logger.info(f'Sending text: {text}')
try:
message_box = WebDriverWait(self.driver, 1).until(
lambda x: x.find_element(By.CLASS_NAME, 'slateTextArea-1Mkdgw'))
except TimeoutException:
self.logger.warning("Discord may have crashed, refreshing page")
self.refresh()
return self.send_text(text)
self.actions.click(on_element=message_box)
for char in text:
self.actions.key_down(char)
self.actions.key_up(char)
self.actions.key_down(Keys.ENTER)
self.actions.key_up(Keys.ENTER)
self.actions.perform()
def react_emoji(self, emoji: str, message_id: int):
self.logger.info(f'Attempting to click emoji: {emoji}')
xpath = f"//div[@id='chat-messages-{message_id}']//div[@aria-label='{emoji}, press to react']"
try:
# Get div containing emoji
emoji_div = WebDriverWait(self.driver, 7).until(lambda x: x.find_element(By.XPATH, xpath))
# Get current count
count = int(emoji_div.find_element(By.XPATH, "//div[@class='reactionCount-2mvXRV']").text)
# Click emoji
# WebElement.click() breaks for some reason, use javascript instead
self.driver.execute_script('arguments[0].click();', emoji_div)
# Check new count
try:
WebDriverWait(self.driver, 1) \
.until_not(lambda x:
int(x.find_element(By.XPATH,
f"{xpath}//div[@class='reactionCount-2mvXRV']").text) > count)
except TimeoutException: # No increase in count
self.logger.warning('Emoji was found, but unsuccessfully clicked')
raise TimeoutError
else:
self.logger.info('Emoji successfully clicked')
except TimeoutException or NoSuchElementException:
self.logger.critical('Unable to find the emoji to click')
raise TimeoutError
def add_heart(self):
# Just type it.
self.send_text(f'+:heart:')
def roll(self, count: int):
"""
Rolls for count number of times.
"""
for _ in range(count):
self.send_text(f'{config.COMMAND_PREFIX}{config.ROLL_COMMAND}')
time.sleep(3) # Sleep for 3 seconds between roll commands
def refresh(self):
self.driver.refresh()
WebDriverWait(self.driver, 10).until(
lambda x: x.find_element(By.CLASS_NAME, 'slateTextArea-1Mkdgw'))
def close(self):
self.driver.close()