-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathfarmer.py
349 lines (328 loc) · 15.4 KB
/
farmer.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
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
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
# farmer.py - twitch-drop-farmer - maxtheaxe
# I just wanna know if it's really this easy
# code taken in part from my client built for zoom.rip
from selenium import webdriver
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.common.action_chains import ActionChains
from selenium.webdriver.common.keys import Keys
# from bs4 import BeautifulSoup as bs
import sys
import re
import time
import random
import signal
import os
from os import path
import inspect
from fake_useragent import UserAgent
import csv
from win10toast import ToastNotifier
# set path to chrome driver for packaging purposes
# ref: https://stackoverflow.com/questions/41030257/is-there-a-way-to-bundle-a-binary-file-such-as-chromedriver-with-a-single-file
current_folder = os.path.realpath(os.path.abspath(os.path.split(inspect.getfile(inspect.currentframe() ))[0]))
# launch() - launch sequence to get driver started, logged in, and prepared to work
def launch(username, password, headless = False, verbose = False, proxy = None):
# print("\tLaunching a bot...\n")
driver = start_driver(headless, proxy) # start the driver and store it (will be returned)
if proxy != None: # if proxies, are given, grab the creds for 'em
proxy_auth(driver) # make sure everything looks okay for bot
# driver.get("https://www.twitch.tv/fl0m") # open first stream
find_dropper(driver) # find a good stream to watch
change_settings(driver) # make it use as little resources as possible
login(driver, username, password, verbose) # log into the room with the room link and name
toaster = ToastNotifier()
toaster.show_toast("Twitch Drop Farmer","Please enter your confirmation code.")
input("\n\tHit Enter to continue...") # wait for user prompt to resume
# print("\t", bot_name, "successfully launched.\n")
return driver # return driver so it can be stored and used later
# start_driver() - starts the webdriver and returns it
# reference: https://tarunlalwani.com/post/selenium-change-user-agent-different-browsers/
# reference: https://groups.google.com/forum/?hl=en-GB#!topic/selenium-users/ZANuzTA2VYQ
# reference: https://stackoverflow.com/questions/11450158/how-do-i-set-proxy-for-chrome-in-python-webdriver
# reference: https://stackoverflow.com/questions/48454949/how-do-i-create-a-random-user-agent-in-python-selenium
def start_driver(headless = False, proxy = None):
# set path to chrome driver for packaging purposes
chromedriver = os.path.join(current_folder,"chromedriver.exe")
# geckodriver = os.path.join(current_folder,"geckodriver.exe") # ff
# setup webdriver settings
options = webdriver.ChromeOptions()
# options = webdriver.FirefoxOptions() # ff
# profile = webdriver.FirefoxProfile() # ff
# check if extensions are downloaded
if path.exists("ublock_origin.crx") and (path.exists("proxy_auto_auth.crx")):
options.add_extension("ublock_origin.crx") # add ublock origin to reduce impact, block stuff
options.add_extension("proxy_auto_auth.crx") # add auto auth for proxies
else:
print("\n\tHang on! You may have forgotten to set up the extensions. See:\n\n\thttps://github.com/maxtheaxe/twitch-drop-farmer/issues/2#issuecomment-868561688\n")
exit()
# other settings
options.headless = headless # headless or not, passed as arg
options.add_experimental_option('excludeSwitches', ['enable-logging']) # chrome only maybe
# make window size bigger
# options.add_argument("--window-size=1600,1200")
# add proxy if one is given
if (proxy != None):
# IP:PORT or HOST:PORT
options.add_argument('--proxy-server=%s' % proxy)
print("\n\tUsing proxy: ", proxy)
# profile.set_preference("network.proxy.type", 1) # ff
# profile.set_preference("network.proxy.http", proxy) # ff
# profile.set_preference("network.proxy.http_port", port) # ff
# add a fake user agent
# user_agent = UserAgent().random # generate random useragent
# print("user agent: ", user_agent) # check what it is
# options.add_argument(f'user-agent={user_agent}')
# profile.set_preference("general.useragent.override", user_agent) # ff
# update profile (ff)
# profile.update_preferences() # ff
# start webdriver and return it
# return webdriver.Firefox(firefox_profile = profile, options = options, executable_path = geckodriver)
return webdriver.Chrome(options = options, executable_path = chromedriver)
# proxy_auth() - auth proxy at start
# reference: https://stackoverflow.com/questions/46330245/userpass-proxies-with-selenium
def proxy_auth(driver):
# open proxy credentials file in read mode
with open("proxy-auth.txt", 'r') as file_handle:
# read file content into list (each detail is on a line)
auth = file_handle.readlines()
# navigate to extension settings
driver.get("chrome-extension://ggmdpepbjljkkkdaklfihhngmmgmpggp/options.html")
# type in details
driver.find_element_by_id("login").send_keys(auth[0]) # username
driver.find_element_by_id("password").send_keys(auth[1]) # password
driver.find_element_by_id("retry").clear()
driver.find_element_by_id("retry").send_keys("2") # retry times
# save it
driver.find_element_by_id("save").click()
# check it
driver.get("https://www.whatsmyip.org/") # navigate to helpful website
# stop for a sec so I can see
time.sleep(1)
return # move on
# login() - logs into the twitch acc
# reference: https://crossbrowsertesting.com/blog/test-automation/automate-login-with-selenium/
# reference: https://stackoverflow.com/questions/19035186/how-to-select-element-using-xpath-syntax-on-selenium-for-python
# future: add password support (for locked rooms)
def login(driver, username, password, verbose = False):
# print("\tLogging in...\n")
web_link = "https://www.twitch.tv/login" # twitch login page
try: # try opening the given link, logging in
driver.get(web_link) # open zoom meeting login page
driver.find_element_by_id('login-username').send_keys(username) # enter username
# enter password
driver.find_element_by_id('password-input').send_keys(password)
# find and click login button
driver.find_element_by_xpath(
"//div[@class='tw-mg-t-2']//button[@data-a-target='passport-login-button']").click()
except: # bad link, try again
print("\tError: Login Failed.\n")
sys.exit()
try: # wait and make sure we're logged in, loaded into the room
wait = WebDriverWait(driver, 10)
element = wait.until(EC.element_to_be_clickable((By.CLASS_NAME, 'facebook-connect-button')))
if EC.visibility_of(driver.find_element_by_class_name("facebook-connect-button")):
if verbose:
print("\tSuccessfully logged in.\n")
except: # something went wrong, we weren't able to load into the room
print("\tError: Login Failed.\n")
sys.exit()
# grab_creds() - grabs credentials from file
def grab_creds():
# read in data and add it to 2d array data
data = list(csv.reader(open("account-combos.csv")))
# chop off the headers
data = data[1:]
return data
# read_proxies() - feed a list of proxies from a file
# ref: https://codippa.com/how-to-read-a-file-line-by-line-into-a-list-in-python/
def read_proxies(file_name):
# open file in read mode
with open(file_name, 'r') as file_handle:
# read file content into list broken up by line
lines = file_handle.readlines()
return lines # return list of proxies
# cavalry() - brings the "cavalry," by launching bots with creds from list
def cavalry(num_bots = 1, headless = False, verbose = False, proxy = None):
driver_list = [] # create list for storing newly-created bots
creds = grab_creds() # store creds in 2d array
for i in range(num_bots): # loop for as many times as num bots desired
# set bot_name by popping name from attendance list (name_options)
username = creds[i][0] # get username from first element on each sublist
password = creds[i][1] # get password from second element on each sublist
# if proxy is given
if proxy != None:
# launch another driver/client with the given info
driver = launch(username, password, headless, verbose, proxy[i])
else: # otherwise, don't use a proxy
# launch another driver/client with the given info
driver = launch(username, password, headless, verbose)
# store the newly-created bot in our list of drivers
driver_list.append(driver)
return driver_list # return the list of newly-created drivers
# enter_stream() - enters a given stream on all the bots
def enter_stream(stream_link):
global bot_list # bring in the bot list
# for all bots stored in master list
for i in range(len(bot_list)):
bot_list[i].get(stream_link) # navigate to the stream
return
# change_settings() - change settings for max performance
def change_settings(driver):
try: # try to click right away
# hover over video player
video_player = driver.find_element_by_class_name('video-player__default-player')
# build new action chain to do so
action = ActionChains(driver)
# hover over own name on participants list
action.move_to_element(video_player).perform()
# find the gear cog and click
# gear = driver.find_element_by_id('7f0643a6c83f8d0cdefcea447825c365').click()
gear = driver.find_element_by_xpath(
"//div[@data-test-selector='settings-menu-button__animate-wrapper']//button[@data-a-target='player-settings-button']")
gear.click()
# find quality dropdown and click
dropdown = driver.find_element_by_xpath(
"//div[contains(text(), 'Quality')]")
dropdown.click()
# find 160p quality setting and click
driver.find_element_by_xpath(
"//div[@data-a-target='player-settings-submenu-quality-option']//div[contains(text(), '160p')]").click()
# next, make the window much smaller (rm bc I'm not sure it really changes impact)
# driver.set_window_size(516, 300)
except: # try again if it wasn't clickable (again, go away)
time.sleep(5) # wait
change_settings(driver) # call again
time.sleep(2) # wait, let em save
return
# find_dropper() - finds a live stream doing drops
def find_dropper(driver):
# find a new stream (navigate to twitch tag "drops enabled")
driver.get("https://www.twitch.tv/directory/all/tags/c2542d6d-cd10-4532-919b-3d19f30a768b")
try: # try to click it right away
# then select the first stream in the directory
new_stream = driver.find_element_by_xpath(
"//div[@data-target='directory-first-item']//a[@data-a-target='preview-card-title-link']")
# and click on the link, navigating to the new stream
new_stream.get_attribute("href").click()
except: # if it isn't clickable (sometimes takes a sec to load properly)
# too lazy to actually check and wait, go away Lrrr
# print("\tFailed. Trying again, please wait...\n")
time.sleep(5)
# then select the first stream in the directory
new_stream = driver.find_element_by_xpath(
"//div[@data-target='directory-first-item']//a[@data-a-target='preview-card-title-link']")
# and click on the link, navigating to the new stream
new_stream.click()
return
# check_error() - checks to see if stream has an error; if so, refreshes
def check_error(driver):
try: # attempt to look for an error
driver.find_element_by_xpath(
"//div[@data-a-target='player-overlay-content-gate']//p[contains(text(), 'Error')]")
# an error must've been found if we got this far, so we need to refresh browser
driver.refresh()
except: # if it throws an exception (assuming I've written this right), no error
print("\n\tNo error found!") # let user know
return
# check_live() - checks if the stream is live or not, finds new one if so
def check_live(driver):
try: # attempt to look for live indicator
driver.find_element_by_xpath(
"//div[@data-target='channel-header-left']//p[contains(text(), 'Live')]")
# if we got this far, it found it, and thus the channel is live; do nothing
except: # if not, find a new stream and switch to it
find_dropper(driver)
return
# maintenance() - perform maintenance on bot to ensure it keeps farming
def maintenance(driver):
print("\tPerforming maintenance...\n")
# next, check if the stream is live, and find a new one if not
check_live(driver)
# check if there's an error displayed (and refresh if so)
check_error(driver)
print("\tMaintenance complete.\n")
return
# group_maintenance() - performs maintenance on list of bots
def group_maintenance(driver_list):
# for each bot in list
for i in range(len(driver_list)):
# run maintenance
maintenance(driver_list[i])
return
# group_maintenance() - finds a dropper for all bots
def find_streams(driver_list):
# for each bot in list
for i in range(len(driver_list)):
# run maintenance
find_dropper(driver_list[i])
return
# get_driver_info() - export driver info in case restart is desired
# reference: https://stackoverflow.com/questions/8344776/can-selenium-interact-with-an-existing-browser-session
def get_driver_info(driver):
url = driver.command_executor._url # "http://127.0.0.1:60622/hub"
session_id = driver.session_id # '4e167f26-dc1d-4f51-a207-f761eaf73c31'
return [url, session_id] # return both things in a list
# save_drivers() - gets and saves driver info for every driver
def save_drivers(driver_list):
# make new list to store info
saved_info = []
# loop over all drivers
for i in range(len(driver_list)):
# append info for each driver to main list as sublist
save_info.append(get_driver_info(driver_list[i]))
# return all saved info as 2d list
return saved_info
# signal_handler() - handles closing the program, to ensure all drivers are quit properly
# reference: https://www.devdungeon.com/content/python-catch-sigint-ctrl-c
def signal_handler(signal_received, frame):
global bot_list # I've never used global variables with python, so idrk what I'm doin
# Handle any cleanup here
print("\n\tClosing all bots, please wait...\n")
# for all bots stored in master list
for i in range(len(bot_list)):
bot_list[i].quit() # quit each bot window
print("\tAll done! Hopefully you got some drops!\n")
sys.exit(0)
def main(argv):
print("\n\t--- Twitch Drop Farmer ---\n")
driver_list = [] # store the bots so we can close 'em later
main_driver = launch("xxx", False)
driver_list.append(main_driver) # store the main one in the list
go_dark(driver[0])
time.sleep(60)
if __name__ == '__main__':
print("\n\t--- Twitch Drop Farmer by Max ---\n")
# main(sys.argv)
num_bots = int(input("\tHow many accounts do you want to idle?\n\t(Type it in and press Enter)\n\n\t"))
headless = False
verbose = False
# check if proxy list exists, otherwise we won't use proxies
if os.path.exists("proxy-list.txt"):
proxy = read_proxies("proxy-list.txt") # read in proxies from file (new one on every line)
# if the user is using more than one bot, let 'em know it's a bad idea
elif (num_bots > 1):
print("\tWarning: Running multiple bots on a single IP Address will have consequences, I recommend against it.")
proxy = None # set proxy as none so we don't have to do diff calls
else:
proxy = None # set proxy as none so we don't have to do diff calls
print("\n\tStarting idling...")
bot_list = cavalry(num_bots, headless, verbose, proxy) # store the bots in a list for easy closing later
# Tell Python to run the handler() function when SIGINT is recieved
signal.signal(signal.SIGINT, signal_handler)
print("\n\tUse Control + C to close all bots.") # print instructions
# wait 25 min for login code (this was a workaround for when I had issues getting codes)
# time.sleep(1500)
while True:
# run maintenance every 20 minutes
try:
group_maintenance(bot_list)
except:
print("\tCheck your internet speed--it may not be fast enough to run this.")
time.sleep(1200)
# ask for a stream link
# selected_stream = input("\tPaste a stream link and hit enter.\n")
# tell all the bots to enter that stream
# enter_stream(selected_stream)
pass