Skip to content

Instantly share code, notes, and snippets.

@ourway
Forked from kyoto-cheng/data_scraping.py
Created April 21, 2023 03:27
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save ourway/25110f1d459120852f8ecfa390c2e2bc to your computer and use it in GitHub Desktop.
Save ourway/25110f1d459120852f8ecfa390c2e2bc to your computer and use it in GitHub Desktop.
from selenium.common.exceptions import NoSuchElementException, ElementClickInterceptedException
from selenium import webdriver
import time
import pandas as pd
def get_jobs(keyword, num_jobs, verbose):
'''Gathers jobs as a dataframe, scraped from Glassdoor'''
# Initializing the webdriver
options = webdriver.ChromeOptions()
# Uncomment the line below if you'd like to scrape without a new Chrome window every time.
#options.add_argument('headless')
# Change the path to where chromedriver is in your home folder.
driver = webdriver.Chrome(executable_path="path_to_chromedriver", options=options)
driver.set_window_size(1120, 1000)
url = 'https://www.glassdoor.com/Job/jobs.htm?sc.keyword="' + keyword + '"&locT=C&locId=1147401&locKeyword=San%20Francisco,%20CA&jobType=all&fromAge=-1&minSalary=0&includeNoSalaryJobs=true&radius=100&cityId=-1&minRating=0.0&industryId=-1&sgocId=-1&seniorityType=all&companyId=-1&employerSizes=0&applicationType=0&remoteWorkType=0'
driver.get(url)
# Define a global variable so even if error happens the progress can still be saved
global jobs
jobs = []
# Let the page load. Change this number based on your internet speed.
# Maybe add extra sleeping at the steps you need for more loading time.
time.sleep(5)
# Click on the first job & Test for the "Sign Up" prompt and get rid of it.
driver.find_element_by_xpath("//*[@id='MainCol']/div[1]/ul/li[1]").click()
time.sleep(1)
# Clicking on the Close X button to close the "Sign Up" prompt.
driver.find_element_by_xpath('//*[@id="JAModal"]/div/div[2]/span').click()
time.sleep(5)
# Clicking on the "Search" button to search for the job postings based on keywords entered.
driver.find_element_by_xpath('//*[@id="scBar"]/div/button/span').click()
time.sleep(10)
while len(jobs) < num_jobs:
# Going through each job in this page
job_buttons = driver.find_elements_by_xpath("//*[@id='MainCol']/div[1]/ul/li")
for job_button in job_buttons:
print("Progress: {}".format("" + str(len(jobs)) + "/" + str(num_jobs)))
if len(jobs) >= num_jobs:
# When the number of jobs collected has reached the number we set.
break
job_button.click()
time.sleep(5)
collected_successfully = False
while not collected_successfully:
try:
time.sleep(5)
company_name = driver.find_element_by_xpath('//*[@id="JDCol"]/div/article/div/div[1]/div/div/div[1]/div[3]/div[1]/div[1]').text
location = driver.find_element_by_xpath('//*[@id="JDCol"]/div/article/div/div[1]/div/div/div[1]/div[3]/div[1]/div[3]').text
job_title = driver.find_element_by_xpath('//*[@id="JDCol"]/div/article/div/div[1]/div/div/div[1]/div[3]/div[1]/div[2]').text
job_description = driver.find_element_by_xpath('//div[@class="jobDescriptionContent desc"]').text
collected_successfully = True
except:
time.sleep(5)
collected_successfully = True
try:
salary_estimate = driver.find_element_by_xpath('//*[@id="JDCol"]/div/article/div/div[1]/div/div/div[1]/div[3]/div[1]/div[4]/span').text
except NoSuchElementException:
# You need to set a "not found value. It's important."
salary_estimate = -1
try:
rating = driver.find_element_by_xpath('//*[@id="JDCol"]/div/article/div/div[1]/div/div/div[1]/div[3]/div[1]/div[1]/span').text
except NoSuchElementException:
# You need to set a "not found value. It's important."
rating = -1
#Printing for debugging
if verbose:
print("Job Title: {}".format(job_title))
print("Salary Estimate: {}".format(salary_estimate))
print("Job Description: {}".format(job_description))
print("Rating: {}".format(rating))
print("Company Name: {}".format(company_name))
print("Location: {}".format(location))
#Going to the Company tab...
time.sleep(5)
try:
driver.find_element_by_xpath('//div[@data-item="tab" and @data-tab-type="overview"]').click()
time.sleep(5)
try:
size = driver.find_element_by_xpath('//*[@id="EmpBasicInfo"]/div[1]/div/div[1]/span[2]').text
except NoSuchElementException:
size = -1
try:
founded = driver.find_element_by_xpath('//*[@id="EmpBasicInfo"]/div[1]/div/div[2]/span[2]').text
except NoSuchElementException:
founded = -1
try:
type_of_ownership = driver.find_element_by_xpath('//*[@id="EmpBasicInfo"]/div[1]/div/div[3]/span[2]').text
except NoSuchElementException:
type_of_ownership = -1
try:
industry = driver.find_element_by_xpath('//*[@id="EmpBasicInfo"]/div[1]/div/div[4]/span[2]').text
except NoSuchElementException:
industry = -1
try:
sector = driver.find_element_by_xpath('//*[@id="EmpBasicInfo"]/div[1]/div/div[5]/span[2]').text
except NoSuchElementException:
sector = -1
try:
revenue = driver.find_element_by_xpath('//*[@id="EmpBasicInfo"]/div[1]/div/div[6]/span[2]').text
except NoSuchElementException:
revenue = -1
# Rarely, some job postings do not have the "Company" tab.
except NoSuchElementException:
size = -1
founded = -1
type_of_ownership = -1
industry = -1
sector = -1
revenue = -1
if verbose:
print("Size: {}".format(size))
print("Founded: {}".format(founded))
print("Type of Ownership: {}".format(type_of_ownership))
print("Industry: {}".format(industry))
print("Sector: {}".format(sector))
print("Revenue: {}".format(revenue))
print("@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@")
jobs.append({
"Job Title" : job_title,
"Salary Estimate" : salary_estimate,
"Job Description" : job_description,
"Rating" : rating,
"Company Name" : company_name,
"Location" : location,
"Size" : size,
"Founded" : founded,
"Type of ownership" : type_of_ownership,
"Industry" : industry,
"Sector" : sector,
"Revenue" : revenue
})
# Clicking on the "next page" button
try:
driver.find_element_by_xpath('//*[@id="FooterPageNav"]/div/ul/li[7]/a/span').click()
time.sleep(10)
except NoSuchElementException:
print("Scraping terminated before reaching target number of jobs. Needed {}, got {}.".format(num_jobs, len(jobs)))
break
#This line converts the dictionary object into a pandas DataFrame.
return pd.DataFrame(jobs)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment