This repository has been archived by the owner on Mar 1, 2020. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy patheasyapplybot.py
231 lines (184 loc) · 7.53 KB
/
easyapplybot.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
import time
import random
from selenium import webdriver
from selenium.webdriver.chrome.options import Options
from bs4 import BeautifulSoup
import pyautogui
from tkinter import filedialog, Tk
import os
class EasyApplyBot:
MAX_APPLICATIONS = 30
def __init__(self, language):
self.language = language
self.options = self.browser_options()
self.browser = webdriver.Chrome(chrome_options=self.options)
self.start_linkedin()
def browser_options(self):
options = Options()
options.add_argument("--start-maximized")
options.add_argument("--ignore-certificate-errors")
options.add_argument("user-agent=Chrome/51.0.2704.79 Safari/537.36 Edge/14.14393")
return options
def start_linkedin(self):
self.browser.get("https://linkedin.com/uas/login")
def wait_for_login(self):
if language == "en":
title = "Sign In to LinkedIn"
elif language == "pt":
title = "Entrar no LinkedIn"
time.sleep(1)
while True:
if self.browser.title != title:
print("\nStarting LinkedIn bot\n")
break
else:
time.sleep(1)
print("\nPlease Login to your LinkedIn account\n")
def fill_data(self):
self.browser.set_window_size(0, 0)
self.browser.set_window_position(2000, 2000)
os.system("reset")
position = input('Enter the desired job title : ')
self.position = position.replace(" ", "%20")
print("Location codes: \n[1] GLOBAL\n[2] Country\n[3] State\n[4] City")
location_code = input('Enter the location code: ')
if location_code == "1":
location = "Worldwide"
elif location_code == "2":
location = input('Enter the country name: ')
elif location_code == "3":
location = input('Enter the state name: ')
elif location_code == "4":
location = input('Enter the city name: ')
self.location = "&location=" + location.replace(" ", "%20") + "&sortBy=DD"
print("\nPlease select your curriculum\n")
time.sleep(1)
root = Tk()
self.resumeloctn = filedialog.askopenfilename(parent=root, initialdir="/",
title='Please select your curriculum')
root.destroy()
def start_apply(self):
self.wait_for_login()
self.fill_data()
self.applications_loop()
def applications_loop(self):
count_application = 0
count_job = 0
jobs_per_page = 0
os.system("reset")
self.browser.set_window_position(0, 0)
self.browser.maximize_window()
self.browser, _ = self.next_jobs_page(jobs_per_page)
while count_application < self.MAX_APPLICATIONS:
# sleep to make sure everything loads, add random to make us look human.
time.sleep(random.uniform(3.5, 6.9))
page = BeautifulSoup(self.browser.page_source, 'lxml')
jobs = self.get_job_links(page)
if not jobs:
print("Jobs not found")
break
for job in jobs:
count_job += 1
job_page = self.get_job_page(job)
if self.got_easy_apply(job_page):
string_easy = "* has Easy Apply Button"
xpath = self.easy_apply_xpath()
self.click_button(xpath)
self.send_resume()
count_application += 1
else:
string_easy = "* Doesn't have Easy Apply Button"
position_number = str(count_job + jobs_per_page)
print(f"\nPosition {position_number}:\n {self.browser.title} \n {string_easy} \n")
if count_job == len(jobs):
jobs_per_page = jobs_per_page + 25
count_job = 0
print("Going to next jobs page !")
self.avoid_lock()
self.browser, jobs_per_page = self.next_jobs_page(jobs_per_page)
self.finish_apply()
def get_job_links(self, page):
links = []
for link in page.find_all('a'):
url = link.get('href')
if url:
if '/jobs/view' in url:
links.append(url)
return set(links)
def get_job_page(self, job):
root = 'www.linkedin.com'
if root not in job:
job = 'https://www.linkedin.com'+job
self.browser.get(job)
self.job_page = self.load_page(sleep=0.5)
return self.job_page
def got_easy_apply(self, page):
button = page.find("button", class_="jobs-s-apply__button js-apply-button")
return len(str(button)) > 4
def get_easy_apply_button(self):
button_class = "jobs-s-apply--top-card jobs-s-apply--fadein inline-flex mr2 jobs-s-apply ember-view"
button = self.job_page.find("div", class_=button_class)
return button
def easy_apply_xpath(self):
button = self.get_easy_apply_button()
button_inner_html = str(button)
list_of_words = button_inner_html.split()
next_word = [word for word in list_of_words if "ember" in word and "id" in word]
ember = next_word[0][:-1]
xpath = '//*[@'+ember+']/button'
return xpath
def click_button(self, xpath):
triggerDropDown = self.browser.find_element_by_xpath(xpath)
time.sleep(0.5)
triggerDropDown.click()
time.sleep(1.5)
def send_resume(self):
self.browser.find_element_by_xpath('//*[@id="file-browse-input"]').send_keys(self.resumeloctn)
submit_button = None
time.sleep(1)
while not submit_button:
if language == "en":
submit_button = self.browser.find_element_by_xpath("//*[contains(text(), 'Submit application')]")
elif language == "pt":
submit_button = self.browser.find_element_by_xpath("//*[contains(text(), 'Enviar candidatura')]")
submit_button.click()
time.sleep(random.uniform(1.5, 2.5))
def load_page(self, sleep=1):
scroll_page = 0
while scroll_page < 4000:
self.browser.execute_script("window.scrollTo(0,"+str(scroll_page)+" );")
scroll_page += 200
time.sleep(sleep)
if sleep != 1:
self.browser.execute_script("window.scrollTo(0,0);")
time.sleep(sleep * 3)
page = BeautifulSoup(self.browser.page_source, "lxml")
return page
def avoid_lock(self):
x, _ = pyautogui.position()
pyautogui.moveTo(x+200, None, duration=1.0)
pyautogui.moveTo(x, None, duration=0.5)
pyautogui.keyDown('ctrl')
pyautogui.press('esc')
pyautogui.keyUp('ctrl')
time.sleep(0.5)
pyautogui.press('esc')
def next_jobs_page(self, jobs_per_page):
self.browser.get(
"https://www.linkedin.com/jobs/search/?f_LF=f_AL&keywords=" +
self.position + self.location + "&start="+str(jobs_per_page))
self.avoid_lock()
self.load_page()
return (self.browser, jobs_per_page)
def finish_apply(self):
self.browser.close()
if __name__ == '__main__':
print("\nEasy Apply Bot\n")
languages = ["pt", "en"]
language = None
while language not in languages:
language = input("LinkedIn language [pt/en]: ")
if language not in languages:
print("Please, enter a valid language!\n")
bot = EasyApplyBot(language)
bot.start_apply()