diff --git a/README.md b/README.md
new file mode 100644
index 0000000..848ba0f
--- /dev/null
+++ b/README.md
@@ -0,0 +1,42 @@
+# UofT PEY Co-op Job Postings Analysis
+
+This repository contains the code for parsing approximately 1.8k HTML pages of UofT PEY co-op job postings (from September 2023 to May 2024) to a single `sqlite3` database file.
+
+## Releases
+
+See releases to access the latest `sqlite3` database with PEY Co-op job postings.
+
+## File Structure
+
+- `parse_to_db.py`: main Python script used for parsing the HTML pages.
+- `requirements.txt`: required imports
+
+## Getting Started
+
+To get started with this project, clone the repository and install the necessary Python dependencies.
+
+```bash
+git clone https://github.com/username/uoft-pey-coop-job-postings-analysis.git
+cd uoft-pey-coop-job-postings-analysis
+pip install -r requirements.txt
+```
+
+## Usage
+
+To parse the HTML pages and store the data in a database, run the `parse_to_db.py` script.
+
+```bash
+python parse_to_db.py
+```
+
+## Additional Information
+
+The `index.qmd` file contains additional information about the job postings. This includes details such as the job title, company, location, and job description. This information is used to enrich the data obtained from the HTML pages.
+
+## Contributing
+
+Contributions are welcome! Please read the contributing guidelines before making any changes.
+
+## License
+
+This project is licensed under the MIT License. See the LICENSE file for more details.
\ No newline at end of file
diff --git a/parse_to_db.py b/parse_to_db.py
new file mode 100644
index 0000000..47cecbe
--- /dev/null
+++ b/parse_to_db.py
@@ -0,0 +1,164 @@
+import argparse
+from bs4 import BeautifulSoup
+import re
+import os
+import sqlite3
+from tqdm import tqdm
+import logging
+
+def extract_job_id_from_html(soup):
+ # Try to find job ID in a
tag with the specific class
+ header_tag = soup.find('h1', class_='h3 dashboard-header__profile-information-name mobile--small-font color--font--white margin--b--s')
+ if header_tag:
+ header_text = header_tag.get_text(strip=True)
+ match = re.match(r'^(\d+)', header_text)
+ if match:
+ return match.group(1)
+
+ # If not found, try to find an tag containing the words "Job ID"
+ job_id_tag = soup.find('h1', string=re.compile(r'Job ID', re.IGNORECASE))
+ if job_id_tag:
+ job_id_text = job_id_tag.get_text(strip=True)
+ match = re.search(r'Job ID\s*:\s*(\d+)', job_id_text, re.IGNORECASE)
+ if match:
+ return match.group(1)
+
+ return None
+
+def parse_html_file(filepath, job_posting_date, verbose=False):
+ with open(filepath, 'r', encoding='utf-8') as file:
+ html_content = file.read()
+
+ soup = BeautifulSoup(html_content, 'lxml')
+
+ # Extract the year, month, and day from the job_posting_date string
+ posting_date = job_posting_date.split('_')[0]
+ data = {'postingDate': posting_date}
+ job_id = extract_job_id_from_html(soup)
+ if job_id:
+ data['id'] = job_id
+
+ rows = soup.find_all('tr') # find all table rows
+
+ for row in rows:
+ tds = row.find_all('td') # find all table data cells
+
+ if len(tds) >= 2:
+ label_td = tds[0]
+ label_text = '\n'.join(label_td.stripped_strings).replace(':', '')
+
+ value_td = tds[1]
+ value_text = '\n'.join(value_td.stripped_strings)
+
+ links = value_td.find_all('a')
+ for link in links:
+ url = link.get('href')
+ link_text = link.get_text()
+ value_text = value_text.replace(link_text, f'{link_text} ({url})')
+
+ # Map label_text to corresponding database column
+ column_mapping = {
+ # 'Job ID': 'id',
+ # 'Job Posting Date': 'postingDate',
+ 'Job Title': 'title',
+ 'Organization': 'company',
+ 'Division': 'companyDivision',
+ 'Website': 'companyWebsite',
+ 'Job Location': 'location',
+ 'Job Location Type': 'locationType',
+ 'Number of Positions': 'numPositions',
+ 'Salary': 'salary',
+ 'Start Date': 'startDate',
+ 'End Date': 'endDate',
+ 'Job Function': 'function',
+ 'Job Description': 'description',
+ 'Job Requirements': 'requirements',
+ 'Preferred Disciplines': 'preferredDisciplines',
+ 'Application Deadline': 'applicationDeadline',
+ 'Application Method': 'applicationMethod',
+ 'Application Receipt Procedure': 'applicationReceiptProcedure',
+ 'If by Website, go to': 'applicationReceiptProcedure',
+ 'Additional Application Information': 'applicationDetails',
+ }
+
+ # Check if label_text matches any of the predefined columns
+ if label_text in column_mapping:
+ db_column = column_mapping[label_text]
+ # If key already exists, append the value to it
+ if db_column in data:
+ data[db_column] += f'\n{value_text}'
+ else:
+ data[db_column] = value_text
+
+ return data
+
+def store_data_in_db(data, db_cursor):
+ columns = ', '.join([f'"{key}"' for key in data.keys()])
+ placeholders = ', '.join(['?' for _ in data.values()])
+ sql = f'INSERT INTO "JobPosting" ({columns}) VALUES ({placeholders})'
+ try:
+ db_cursor.execute(sql, tuple(data.values()))
+ except sqlite3.IntegrityError:
+ logging.info("Integrity Error: Skipping row")
+ pass
+
+def create_db_schema(db_cursor):
+ db_cursor.execute('''
+ CREATE TABLE IF NOT EXISTS JobPosting (
+ id INTEGER,
+ postingDate DATE,
+ title TEXT,
+ company TEXT,
+ companyDivision TEXT,
+ companyWebsite TEXT,
+ location TEXT,
+ locationType TEXT,
+ numPositions INTEGER,
+ salary TEXT,
+ startDate TEXT,
+ endDate TEXT,
+ function TEXT,
+ description TEXT,
+ requirements TEXT,
+ preferredDisciplines TEXT,
+ applicationDeadline TEXT,
+ applicationMethod TEXT,
+ applicationReceiptProcedure TEXT,
+ applicationDetails TEXT,
+ PRIMARY KEY(id, postingDate)
+ )
+ ''')
+
+if __name__ == "__main__":
+ logging.basicConfig(filename='run.log', level=logging.INFO, format='%(asctime)s %(message)s')
+
+ parser = argparse.ArgumentParser(description="Parse HTML files in a folder and store data in SQLite DB.")
+ parser.add_argument("-d", "--directory", default=os.getcwd(), help="Path to the directory containing HTML files. Default is the current directory.")
+ parser.add_argument("--db", default=os.path.join(os.getcwd(), "job_postings.db"), help="SQLite database file to store the parsed data. Default is 'job_postings.db' in the directory specified by -d.")
+ parser.add_argument("-v", "--verbose", action="store_true", help="logging.info parsed data.")
+
+ args = parser.parse_args()
+
+ conn = sqlite3.connect(args.db)
+ cursor = conn.cursor()
+ create_db_schema(cursor)
+
+ # Get the list of files
+ files = [os.path.join(dirpath, file) for dirpath, _, files in os.walk(args.directory) for file in files if file.endswith('.html') or file.endswith('.htm')]
+
+ # Create a progress bar
+ with tqdm(total=len(files)) as pbar:
+ for subdir, _, files in os.walk(args.directory):
+ job_posting_date = os.path.basename(subdir)
+ for file in files:
+ if file.endswith('.html') or file.endswith('.htm'):
+ filepath = os.path.join(subdir, file)
+ logging.info(filepath)
+ data = parse_html_file(filepath, job_posting_date, args.verbose)
+ store_data_in_db(data, cursor)
+ # Update the progress bar
+ pbar.update(1)
+
+ conn.commit()
+ conn.close()
+ logging.info("Parsing and storing completed.")
diff --git a/requirements.txt b/requirements.txt
new file mode 100644
index 0000000..5cbdf2f
--- /dev/null
+++ b/requirements.txt
@@ -0,0 +1,2 @@
+beautifulsoup4==4.9.3
+tqdm==4.62.3
\ No newline at end of file