106 lines
3.4 KiB
Python
106 lines
3.4 KiB
Python
from __future__ import annotations
|
|
from dataclasses import dataclass
|
|
import requests
|
|
from requests.adapters import HTTPAdapter
|
|
from urllib3.util.retry import Retry
|
|
import uuid
|
|
from ...exceptions import AuthenticationError
|
|
from .models import Property, ListingType, SiteName
|
|
import json
|
|
|
|
|
|
@dataclass
|
|
class ScraperInput:
|
|
location: str
|
|
listing_type: ListingType
|
|
radius: float | None = None
|
|
mls_only: bool | None = False
|
|
proxy: str | None = None
|
|
last_x_days: int | None = None
|
|
date_from: str | None = None
|
|
date_to: str | None = None
|
|
foreclosure: bool | None = False
|
|
extra_property_data: bool | None = True
|
|
exclude_pending: bool | None = False
|
|
|
|
|
|
class Scraper:
|
|
session = None
|
|
|
|
def __init__(
|
|
self,
|
|
scraper_input: ScraperInput,
|
|
):
|
|
self.location = scraper_input.location
|
|
self.listing_type = scraper_input.listing_type
|
|
|
|
if not self.session:
|
|
Scraper.session = requests.Session()
|
|
retries = Retry(
|
|
total=3, backoff_factor=3, status_forcelist=[429, 403], allowed_methods=frozenset(["GET", "POST"])
|
|
)
|
|
|
|
adapter = HTTPAdapter(max_retries=retries)
|
|
Scraper.session.mount("http://", adapter)
|
|
Scraper.session.mount("https://", adapter)
|
|
Scraper.session.headers.update(
|
|
{
|
|
"auth": f"Bearer {self.get_access_token()}",
|
|
"apollographql-client-name": "com.move.Realtor-apollo-ios",
|
|
}
|
|
)
|
|
|
|
if scraper_input.proxy:
|
|
proxy_url = scraper_input.proxy
|
|
proxies = {"http": proxy_url, "https": proxy_url}
|
|
self.session.proxies.update(proxies)
|
|
|
|
self.listing_type = scraper_input.listing_type
|
|
self.radius = scraper_input.radius
|
|
self.last_x_days = scraper_input.last_x_days
|
|
self.mls_only = scraper_input.mls_only
|
|
self.date_from = scraper_input.date_from
|
|
self.date_to = scraper_input.date_to
|
|
self.foreclosure = scraper_input.foreclosure
|
|
self.extra_property_data = scraper_input.extra_property_data
|
|
self.exclude_pending = scraper_input.exclude_pending
|
|
|
|
def search(self) -> list[Property]: ...
|
|
|
|
@staticmethod
|
|
def _parse_home(home) -> Property: ...
|
|
|
|
def handle_location(self): ...
|
|
|
|
@staticmethod
|
|
def get_access_token():
|
|
device_id = str(uuid.uuid4()).upper()
|
|
|
|
response = requests.post(
|
|
"https://graph.realtor.com/auth/token",
|
|
headers={
|
|
'Host': 'graph.realtor.com',
|
|
'Accept': '*/*',
|
|
'Content-Type': 'Application/json',
|
|
'X-Client-ID': 'rdc_mobile_native,iphone',
|
|
'X-Visitor-ID': device_id,
|
|
'X-Client-Version': '24.21.23.679885',
|
|
'Accept-Language': 'en-US,en;q=0.9',
|
|
'User-Agent': 'Realtor.com/24.21.23.679885 CFNetwork/1494.0.7 Darwin/23.4.0',
|
|
},
|
|
data=json.dumps({
|
|
"grant_type": "device_mobile",
|
|
"device_id": device_id,
|
|
"client_app_id": "rdc_mobile_native,24.21.23.679885,iphone"
|
|
}))
|
|
|
|
data = response.json()
|
|
|
|
if not (access_token := data.get("access_token")):
|
|
raise AuthenticationError(
|
|
"Failed to get access token, use a proxy/vpn or wait a moment and try again.",
|
|
response=response
|
|
)
|
|
|
|
return access_token
|