-
Notifications
You must be signed in to change notification settings - Fork 10
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
san francisco police commission scraper
- Loading branch information
1 parent
8296b63
commit 27ab190
Showing
1 changed file
with
194 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,194 @@ | ||
import time | ||
from pathlib import Path | ||
from typing import List | ||
import re | ||
from urllib.parse import urlparse, parse_qs | ||
|
||
from bs4 import BeautifulSoup, Tag | ||
|
||
from .. import utils | ||
from ..cache import Cache | ||
from ..utils import MetadataDict | ||
|
||
|
||
class Site: | ||
"""Scrape file metadata for the San Francisco Police Commission.""" | ||
|
||
name = "San Francisco Police Commission" | ||
|
||
def __init__( | ||
self, | ||
data_dir: Path = utils.CLEAN_DATA_DIR, | ||
cache_dir: Path = utils.CLEAN_CACHE_DIR, | ||
): | ||
"""Initialize a new instance.""" | ||
self.base_url = "https://www.sf.gov" | ||
self.disclosure_url = f"{self.base_url}/resource/2022/records-released-pursuant-ca-penal-code-ss-8327" | ||
self.data_dir = data_dir | ||
self.cache_dir = cache_dir | ||
self.cache = Cache(cache_dir) | ||
|
||
@property | ||
def agency_slug(self) -> str: | ||
"""Construct the agency slug.""" | ||
mod = Path(__file__) | ||
state_postal = mod.parent.stem | ||
return f"{state_postal}_{mod.stem}" # e.g., ca_san_francisco_pc | ||
|
||
def scrape_meta(self, throttle: int = 0) -> Path: | ||
""" | ||
Gather metadata on downloadable files by following a two-step process: | ||
1. Extract links from main pages. | ||
2. Extract metadata from detail pages. | ||
Args: | ||
throttle (int): Number of seconds to wait between requests. Defaults to 0. | ||
Returns: | ||
Path: Local path of JSON file containing metadata. | ||
""" | ||
# Step 1: Extract links from main pages | ||
main_links = self.get_main_page_links() | ||
|
||
# Step 2: Extract metadata from detail pages | ||
metadata = self.get_detail_page_links(main_links, throttle) | ||
|
||
# Write metadata to a JSON file | ||
outfile = self.data_dir.joinpath(f"{self.agency_slug}.json") | ||
self.cache.write_json(outfile, metadata) | ||
|
||
return outfile | ||
|
||
def get_main_page_links(self) -> List[str]: | ||
""" | ||
Retrieves links from the main page of the site. | ||
Returns: | ||
List[str]: A list of URLs for detailed pages. | ||
""" | ||
main_links = [] | ||
|
||
cache_path = self._download_index_page(self.disclosure_url) | ||
html = self.cache.read(cache_path) | ||
soup = BeautifulSoup(html, "html.parser") | ||
|
||
for link in soup.find_all("a", href=True): | ||
if "RequestArchiveDetails" in link["href"]: | ||
main_links.append( | ||
f"{self.base_url}/{link['href']}" | ||
if not link["href"].startswith("http") | ||
else link["href"] | ||
) | ||
|
||
return main_links | ||
|
||
def get_detail_page_links( | ||
self, main_links: List[str], throttle: int = 0 | ||
) -> List[MetadataDict]: | ||
""" | ||
Extracts detailed metadata from links on the main pages. | ||
Args: | ||
main_links (List[str]): A list of main page URLs. | ||
throttle (int): Number of seconds to wait between requests. | ||
Returns: | ||
List[MetadataDict]: A list of metadata dictionaries for downloadable resources. | ||
""" | ||
metadata = [] | ||
|
||
# Define a regex pattern to match input ids with the format 'rptAttachments_ctlXX_hdnAzureURL' | ||
id_pattern = re.compile(r"^rptAttachments_ctl\d+_hdnAzureURL$") | ||
|
||
for link in main_links: | ||
cache_path = self._download_index_page(link) | ||
html = self.cache.read(cache_path) | ||
soup = BeautifulSoup(html, "html.parser") | ||
|
||
# Extract the case_id from the reference number paragraph (<p>) tag | ||
case_id_tag = soup.find( | ||
"p", style="font-weight: 400; max-width: 75%; font-size: 0.875rem" | ||
) | ||
case_id = case_id_tag.text.strip() if case_id_tag else None | ||
|
||
# Ensure case_id is always a string | ||
case_id = str(case_id) if case_id else "" | ||
|
||
# Find all input tags where the id matches the pattern | ||
input_tags = soup.find_all("input", id=id_pattern) | ||
|
||
# Ensure we process each input tag | ||
for input_tag in input_tags: | ||
value = input_tag.get("value") | ||
if isinstance(value, str): | ||
full_url = value.strip() | ||
if full_url: | ||
# Check if the URL starts with the base domain | ||
if full_url.startswith( | ||
"https://1sanfranciscopd.blob.core.usgovcloudapi.net/" | ||
): | ||
asset_url = full_url | ||
else: | ||
asset_url = ( | ||
"https://1sanfranciscopd.blob.core.usgovcloudapi.net/" | ||
+ full_url.lstrip("/") | ||
) | ||
|
||
# Parse the URL and extract the filename from the query string | ||
parsed_url = urlparse(asset_url) | ||
query_params = parse_qs(parsed_url.query) | ||
|
||
# Get the filename from the 'rscd' parameter | ||
filename = query_params.get("rscd", [None])[0] | ||
|
||
if filename: | ||
# Extract the filename after the 'filename=' part | ||
filename = filename.split("filename=")[-1] | ||
|
||
# Generate a title by removing underscores and .pdf extension | ||
title = filename.replace("_", " ").replace(".pdf", "") | ||
else: | ||
# Default case if filename is not found | ||
filename = asset_url.split("?")[0].rsplit("/", 1)[-1] | ||
title = filename.replace("_", " ").replace(".pdf", "") | ||
|
||
# Set the filename as 'name' | ||
name = ( | ||
filename | ||
if filename | ||
else asset_url.split("?")[0].rsplit("/", 1)[-1] | ||
) | ||
|
||
payload: MetadataDict = { | ||
"asset_url": asset_url, | ||
"case_id": case_id, # Reference No as it appears on the website | ||
"name": name, | ||
"title": title, # Use the formatted title here | ||
"parent_page": link, | ||
} | ||
metadata.append(payload) | ||
|
||
time.sleep(throttle) | ||
|
||
return metadata | ||
|
||
def _download_index_page(self, page_url: str) -> Path: | ||
""" | ||
Download the index page for use for officer involved shootings; | ||
use of force with great bodily injury/death; | ||
& sustained complaints of sexual assault, dishonesty, excessive force, biased conduct, unlawful search or arrest, | ||
and failing to intervene against another officer using excessive force. | ||
Index pages link to child pages containing pdfs. | ||
Returns: | ||
Local path of downloaded file | ||
""" | ||
split_url = page_url.split("/") | ||
# Creates a unique filename using parts of the URL, | ||
# combining the directory and filename, with _index appended. | ||
file_stem = f"{split_url[-4]}_{split_url[-1]}_index" | ||
# Downloads the content from the page_url and stores it locally with the generated file_stem. | ||
cache_path = self.cache.download(file_stem, page_url, "utf-8") | ||
return cache_path |