forked from coding-ai/EasyApply-Linkedin
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
179 lines (147 loc) · 7.74 KB
/
main.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
from selenium import webdriver
from selenium.webdriver.common.keys import Keys
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.common.by import By
from selenium.common.exceptions import TimeoutException, NoSuchElementException, ElementClickInterceptedException, NoSuchElementException
from selenium.webdriver.common.action_chains import ActionChains
import time
import re
import json
class EasyApplyLinkedin:
def __init__(self, data):
"""Parameter initialization"""
self.email = data['email']
self.password = data['password']
self.keywords = data['keywords']
self.location = data['location']
self.driver = webdriver.Chrome(data['driver_path'])
def login_linkedin(self):
"""This function logs into your personal LinkedIn profile"""
# go to the LinkedIn login url
self.driver.get("https://www.linkedin.com/login")
# introduce email and password and hit enter
login_email = self.driver.find_element(By.NAME, 'session_key')
login_email.clear()
login_email.send_keys(self.email)
login_pass = self.driver.find_element(By.NAME, 'session_password')
login_pass.clear()
login_pass.send_keys(self.password)
login_pass.send_keys(Keys.RETURN)
def job_search(self):
"""This function goes to the 'Jobs' section a looks for all the jobs that matches the keywords and location"""
# go to Jobs
jobs_link = self.driver.find_element(By.LINK_TEXT, 'Jobs')
jobs_link.click()
# search based on keywords and location and hit enter
search_keywords = WebDriverWait(self.driver, 20).until(EC.presence_of_element_located((By.CSS_SELECTOR, ".jobs-search-box__text-input[aria-label='Search by title, skill, or company']")))
search_keywords.clear()
search_keywords.send_keys(self.keywords)
search_location = self.driver.find_element(By.CSS_SELECTOR, ".jobs-search-box__text-input[aria-label='City, state, or zip code']")
search_location.clear()
search_location.send_keys(self.location)
# Added this to get it to search for me
time.sleep(2)
search_keywords.send_keys(Keys.RETURN)
def filter(self):
"""This function filters all the job results by 'Easy Apply'"""
# select Easy Apply filter button
all_filters_button = WebDriverWait(self.driver, 20).until(EC.presence_of_element_located((By.XPATH, "//button[contains(@class, 'artdeco-pill artdeco-pill--slate artdeco-pill--2 artdeco-pill--choice ember-view search-reusables__filter-pill-button')]")))
all_filters_button.click()
time.sleep(1)
def find_offers(self):
"""This function finds all the offers through all the pages result of the search and filter"""
# find the total amount of results (if the results are above 24-more than one page-, we will scroll trhough all available pages)
total_results = self.driver.find_element(By.CSS_SELECTOR, ".display-flex.t-normal.t-12.t-black--light.jobs-search-results-list__text")
total_results_int = int(total_results.text.split(' ',1)[0].replace(",",""))
print(total_results_int)
time.sleep(2)
# get results for the first page
current_page = self.driver.current_url
results = self.driver.find_elements(By.CSS_SELECTOR, ".job-card-container.relative")
# for each job add, submits application if no questions asked
for result in results:
hover = ActionChains(self.driver).move_to_element(result)
hover.perform()
titles = result.find_elements(By.CSS_SELECTOR, '.artdeco-entity-lockup__title.ember-view')
for title in titles:
self.submit_apply(title)
# if there is more than one page, find the pages and apply to the results of each page
if total_results_int > 24:
time.sleep(2)
# find the last page and construct url of each page based on the total amount of pages
find_pages = self.driver.find_elements(By.CLASS_NAME, "artdeco-pagination__indicator.artdeco-pagination__indicator--number.ember-view")
total_pages = find_pages[len(find_pages)-1].text
total_pages_int = int(re.sub(r"[^\d.]", "", total_pages))
get_last_page = self.driver.find_element(By.XPATH, "//button[@aria-label='Page "+str(total_pages_int)+"']")
get_last_page.send_keys(Keys.RETURN)
time.sleep(2)
last_page = self.driver.current_url
total_jobs = int(last_page.split('start=',1)[1])
# go through all available pages and job offers and apply
for page_number in range(25,total_jobs+25,25):
self.driver.get(current_page+'&start='+str(page_number))
time.sleep(2)
results_ext = self.driver.find_elements(By.CSS_SELECTOR, ".job-card-container.relative")
for result_ext in results_ext:
hover_ext = ActionChains(self.driver).move_to_element(result_ext)
hover_ext.perform()
titles_ext = result_ext.find_elements(By.CSS_SELECTOR, '.artdeco-entity-lockup__title.ember-view')
for title_ext in titles_ext:
self.submit_apply(title_ext)
else:
self.close_session()
def submit_apply(self,job_add):
"""This function submits the application for the job add found"""
print('You are applying to the position of: ', job_add.text)
job_add.click()
time.sleep(2)
# click on the easy apply button, skip if already applied to the position
try:
in_apply = self.driver.find_element(By.CLASS_NAME, "jobs-apply-button")
in_apply.click()
except NoSuchElementException:
print('You already applied to this job, go to next...')
pass
time.sleep(1)
# try to submit if submit application is available...
try:
submit = self.driver.find_element(By.XPATH, "//button[@aria-label='Submit application']")
submit.send_keys(Keys.RETURN)
done_button = WebDriverWait(self.driver, 10).until(EC.element_to_be_clickable((By.XPATH, "//button[.//span[text()='Done']]")))
done_button.click()
# ... if not available, discard application and go to next
except NoSuchElementException:
print('Not direct application, going to next...')
try:
wait = WebDriverWait(self.driver, 5)
discard = wait.until(EC.element_to_be_clickable((By.XPATH, "//button[@data-test-modal-close-btn]")))
discard.send_keys(Keys.RETURN)
time.sleep(1)
discard_confirm = wait.until(EC.element_to_be_clickable((By.XPATH, "//button[@data-test-dialog-secondary-btn]")))
discard_confirm.send_keys(Keys.RETURN)
time.sleep(1)
except TimeoutException:
print('Could not find the close button, skipping...')
time.sleep(1)
def close_session(self):
"""This function closes the actual session"""
print('End of the session, see you later!')
self.driver.close()
def apply(self):
"""Apply to job offers"""
self.driver.maximize_window()
self.login_linkedin()
time.sleep(5)
self.job_search()
time.sleep(5)
self.filter()
time.sleep(2)
self.find_offers()
time.sleep(2)
self.close_session()
if __name__ == '__main__':
with open('config.json') as config_file:
data = json.load(config_file)
bot = EasyApplyLinkedin(data)
bot.apply()