parent
8cfe056f79
commit
1f717bd9e3
|
@ -104,6 +104,8 @@ Optional
|
||||||
Property
|
Property
|
||||||
├── Basic Information:
|
├── Basic Information:
|
||||||
│ ├── property_url
|
│ ├── property_url
|
||||||
|
│ ├── property_id
|
||||||
|
│ ├── listing_id
|
||||||
│ ├── mls
|
│ ├── mls
|
||||||
│ ├── mls_id
|
│ ├── mls_id
|
||||||
│ └── status
|
│ └── status
|
||||||
|
|
|
@ -46,8 +46,21 @@ class Scraper:
|
||||||
Scraper.session.mount("https://", adapter)
|
Scraper.session.mount("https://", adapter)
|
||||||
Scraper.session.headers.update(
|
Scraper.session.headers.update(
|
||||||
{
|
{
|
||||||
"auth": f"Bearer {self.get_access_token()}",
|
'accept': 'application/json, text/javascript',
|
||||||
"apollographql-client-name": "com.move.Realtor-apollo-ios",
|
'accept-language': 'en-US,en;q=0.9',
|
||||||
|
'cache-control': 'no-cache',
|
||||||
|
'content-type': 'application/json',
|
||||||
|
'origin': 'https://www.realtor.com',
|
||||||
|
'pragma': 'no-cache',
|
||||||
|
'priority': 'u=1, i',
|
||||||
|
'rdc-ab-tests': 'commute_travel_time_variation:v1',
|
||||||
|
'sec-ch-ua': '"Not)A;Brand";v="99", "Google Chrome";v="127", "Chromium";v="127"',
|
||||||
|
'sec-ch-ua-mobile': '?0',
|
||||||
|
'sec-ch-ua-platform': '"Windows"',
|
||||||
|
'sec-fetch-dest': 'empty',
|
||||||
|
'sec-fetch-mode': 'cors',
|
||||||
|
'sec-fetch-site': 'same-origin',
|
||||||
|
'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/127.0.0.0 Safari/537.36',
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -133,6 +133,10 @@ class Advertisers:
|
||||||
@dataclass
|
@dataclass
|
||||||
class Property:
|
class Property:
|
||||||
property_url: str
|
property_url: str
|
||||||
|
|
||||||
|
property_id: str
|
||||||
|
listing_id: str | None = None
|
||||||
|
|
||||||
mls: str | None = None
|
mls: str | None = None
|
||||||
mls_id: str | None = None
|
mls_id: str | None = None
|
||||||
status: str | None = None
|
status: str | None = None
|
||||||
|
|
|
@ -181,11 +181,9 @@ class RealtorScraper(Scraper):
|
||||||
if "source" in result and isinstance(result["source"], dict)
|
if "source" in result and isinstance(result["source"], dict)
|
||||||
else None
|
else None
|
||||||
),
|
),
|
||||||
property_url=(
|
property_url=result["href"],
|
||||||
f"{self.PROPERTY_URL}{property_id}"
|
property_id=property_id,
|
||||||
if self.listing_type != ListingType.FOR_RENT
|
listing_id=result.get("listing_id"),
|
||||||
else f"{self.PROPERTY_URL}M{property_id}?listing_status=rental"
|
|
||||||
),
|
|
||||||
status="PENDING" if is_pending else result["status"].upper(),
|
status="PENDING" if is_pending else result["status"].upper(),
|
||||||
list_price=result["list_price"],
|
list_price=result["list_price"],
|
||||||
list_price_min=result["list_price_min"],
|
list_price_min=result["list_price_min"],
|
||||||
|
@ -469,7 +467,7 @@ class RealtorScraper(Scraper):
|
||||||
}"""
|
}"""
|
||||||
|
|
||||||
variables = {"property_id": property_id}
|
variables = {"property_id": property_id}
|
||||||
response = self.session.post(self.PROPERTY_GQL, json={"query": query, "variables": variables})
|
response = self.session.post(self.SEARCH_GQL_URL, json={"query": query, "variables": variables})
|
||||||
|
|
||||||
data = response.json()
|
data = response.json()
|
||||||
property_details = data["data"]["home"]
|
property_details = data["data"]["home"]
|
||||||
|
|
|
@ -2,6 +2,7 @@ _SEARCH_HOMES_DATA_BASE = """{
|
||||||
pending_date
|
pending_date
|
||||||
listing_id
|
listing_id
|
||||||
property_id
|
property_id
|
||||||
|
href
|
||||||
list_date
|
list_date
|
||||||
status
|
status
|
||||||
last_sold_price
|
last_sold_price
|
||||||
|
|
|
@ -6,6 +6,8 @@ from .exceptions import InvalidListingType, InvalidDate
|
||||||
|
|
||||||
ordered_properties = [
|
ordered_properties = [
|
||||||
"property_url",
|
"property_url",
|
||||||
|
"property_id",
|
||||||
|
"listing_id",
|
||||||
"mls",
|
"mls",
|
||||||
"mls_id",
|
"mls_id",
|
||||||
"status",
|
"status",
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
[tool.poetry]
|
[tool.poetry]
|
||||||
name = "homeharvest"
|
name = "homeharvest"
|
||||||
version = "0.4.2"
|
version = "0.4.3"
|
||||||
description = "Real estate scraping library"
|
description = "Real estate scraping library"
|
||||||
authors = ["Zachary Hampton <zachary@bunsly.com>", "Cullen Watson <cullen@bunsly.com>"]
|
authors = ["Zachary Hampton <zachary@bunsly.com>", "Cullen Watson <cullen@bunsly.com>"]
|
||||||
homepage = "https://github.com/Bunsly/HomeHarvest"
|
homepage = "https://github.com/Bunsly/HomeHarvest"
|
||||||
|
|
|
@ -128,6 +128,7 @@ def test_realtor_bad_address():
|
||||||
location="abceefg ju098ot498hh9",
|
location="abceefg ju098ot498hh9",
|
||||||
listing_type="for_sale",
|
listing_type="for_sale",
|
||||||
)
|
)
|
||||||
|
|
||||||
if len(bad_results) == 0:
|
if len(bad_results) == 0:
|
||||||
assert True
|
assert True
|
||||||
|
|
||||||
|
@ -253,3 +254,29 @@ def test_builder_exists():
|
||||||
|
|
||||||
assert listing is not None
|
assert listing is not None
|
||||||
assert listing["builder_name"].nunique() > 0
|
assert listing["builder_name"].nunique() > 0
|
||||||
|
|
||||||
|
|
||||||
|
def test_phone_number_matching():
|
||||||
|
searches = [
|
||||||
|
scrape_property(
|
||||||
|
location="Phoenix, AZ",
|
||||||
|
listing_type="for_sale",
|
||||||
|
limit=100,
|
||||||
|
),
|
||||||
|
scrape_property(
|
||||||
|
location="Phoenix, AZ",
|
||||||
|
listing_type="for_sale",
|
||||||
|
limit=100,
|
||||||
|
),
|
||||||
|
]
|
||||||
|
|
||||||
|
assert all([search is not None for search in searches])
|
||||||
|
|
||||||
|
#: random row
|
||||||
|
row = searches[0][searches[0]["agent_phones"].notnull()].sample()
|
||||||
|
|
||||||
|
#: find matching row
|
||||||
|
matching_row = searches[1].loc[searches[1]["property_url"] == row["property_url"].values[0]]
|
||||||
|
|
||||||
|
#: assert phone numbers are the same
|
||||||
|
assert row["agent_phones"].values[0] == matching_row["agent_phones"].values[0]
|
||||||
|
|
Loading…
Reference in New Issue