You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
changedetection.io/changedetectionio/processors/restock_diff.py

163 lines
6.8 KiB

from . import difference_detection_processor
9 months ago
from ..html_tools import xpath1_filter as xpath_filter
# xpath1 is a lot faster and is sufficient here
from ..html_tools import extract_json_as_string, has_ldjson_product_info
from copy import deepcopy
from loguru import logger
import hashlib
1 year ago
import re
import urllib3
urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning)
8 months ago
name = 'Re-stock & Price detection for single product pages'
description = 'Detects if the product goes back to in-stock'
1 year ago
class UnableToExtractRestockData(Exception):
def __init__(self, status_code):
# Set this so we can use it in other parts of the app
self.status_code = status_code
return
1 year ago
9 months ago
def get_itemprop_availability(html_content):
"""
`itemprop` is a global attribute
https://developer.mozilla.org/en-US/docs/Web/HTML/Global_attributes/itemprop
https://schema.org/ItemAvailability
<div class="product-offer" itemprop="offers" itemscope="" itemtype="https://schema.org/Offer">
...
<link itemprop="availability" href="https://schema.org/OutOfStock" />
:return:
"""
# Try/prefer the structured data first if it exists
# https://schema.org/ItemAvailability Which strings mean we should consider it in stock?
9 months ago
# Chewing on random content could throw any kind of exception, best to catch it and move on if possible.
8 months ago
import json
9 months ago
8 months ago
# LD-JSON type
value = {'price': None, 'availability': None, 'currency': None}
9 months ago
try:
9 months ago
if has_ldjson_product_info(html_content):
8 months ago
res = extract_json_as_string(html_content.lower(), "json:$..offers", ensure_is_ldjson_info_type=True)
if res:
logger.debug(f"Has 'LD-JSON' - '{value}'")
8 months ago
ld_obj = json.loads(res)
if ld_obj and isinstance(ld_obj, list):
ld_obj = ld_obj[0]
value['price'] = ld_obj.get('price')
value['currency'] = ld_obj['pricecurrency'].upper() if ld_obj.get('pricecurrency') else None
value['availability'] = ld_obj['availability'] if ld_obj.get('availability') else None
9 months ago
9 months ago
except Exception as e:
# This should be OK, we will attempt the scraped version instead
logger.warning(f"Exception getting get_itemprop_availability 'LD-JSON' - {str(e)}")
9 months ago
9 months ago
# Microdata style
8 months ago
if not value.get('price'):
9 months ago
try:
8 months ago
res = xpath_filter("//*[@itemtype='https://schema.org/Offer']//*[@itemprop='availability']/@href", html_content)
if res:
#@todo
logger.debug(f"Has 'Microdata' - '{value}'")
9 months ago
9 months ago
except Exception as e:
# This should be OK, we will attempt the scraped version instead
logger.warning(f"Exception getting get_itemprop_availability 'Microdata' - {str(e)}")
9 months ago
9 months ago
# RDFa style
8 months ago
if not value.get('price'):
try:
8 months ago
res = xpath_filter("//*[@property='schema:availability']/@content", html_content)
# @todo
if res:
logger.debug(f"Has 'RDFa' - '{value}'")
except Exception as e:
9 months ago
# This should be OK, we will attempt the scraped version instead
logger.warning(f"Exception getting get_itemprop_availability 'RDFa' - {str(e)}")
8 months ago
value['availability'] = re.sub(r'(?i)^(https|http)://schema.org/', '', value.get('availability').strip(' "\'').lower()) if value.get('availability') else None
9 months ago
# @todo this should return dict/tuple of instock + price
9 months ago
return value
9 months ago
class perform_site_check(difference_detection_processor):
screenshot = None
xpath_data = None
def run_changedetection(self, uuid, skip_when_checksum_same=True):
# DeepCopy so we can be sure we don't accidently change anything by reference
watch = deepcopy(self.datastore.data['watching'].get(uuid))
if not watch:
raise Exception("Watch no longer exists.")
# Unset any existing notification error
8 months ago
update_obj = {'last_notification_error': False, 'last_error': False, 'in_stock': None, 'restock': None}
self.screenshot = self.fetcher.screenshot
self.xpath_data = self.fetcher.xpath_data
# Track the content type
update_obj['content_type'] = self.fetcher.headers.get('Content-Type', '')
update_obj["last_check_status"] = self.fetcher.get_last_status_code()
8 months ago
itemprop_availability = get_itemprop_availability(html_content=self.fetcher.content)
if itemprop_availability.get('price') or itemprop_availability.get('availability'):
# Store for other usage
update_obj['restock'] = itemprop_availability
9 months ago
9 months ago
# @todo: Configurable?
8 months ago
if any(substring.lower() in itemprop_availability['availability'].lower() for substring in [
'instock',
'instoreonly',
'limitedavailability',
'onlineonly',
'presale']
):
update_obj['restock']['in_stock'] = True
else:
8 months ago
update_obj['restock']['in_stock'] = False
8 months ago
# Used for the change detection, we store the real data separately, in the future this can implement some min,max threshold
# @todo if price is None?
self.fetcher.instock_data = f"{itemprop_availability.get('availability')} - {itemprop_availability.get('price')}"
elif self.fetcher.instock_data:
# 'Possibly in stock' comes from stock-not-in-stock.js when no string found above in the metadata of the HTML
update_obj['restock']['in_stock'] = True if self.fetcher.instock_data == 'Possibly in stock' else False
logger.debug(f"Restock - using scraped browserdata - Watch UUID {uuid} restock check returned '{self.fetcher.instock_data}' from JS scraper.")
9 months ago
if not self.fetcher.instock_data:
raise UnableToExtractRestockData(status_code=self.fetcher.status_code)
# Main detection method
9 months ago
fetched_md5 = hashlib.md5(self.fetcher.instock_data.encode('utf-8')).hexdigest()
# The main thing that all this at the moment comes down to :)
changed_detected = False
logger.debug(f"Watch UUID {uuid} restock check - Previous MD5: {watch.get('previous_md5')}, Fetched MD5 {fetched_md5}")
8 months ago
if watch['restock'].get('in_stock') != update_obj['restock'].get('in_stock'):
# Yes if we only care about it going to instock, AND we are in stock
8 months ago
if watch.get('in_stock_only') and update_obj['restock']['in_stock']:
changed_detected = True
if not watch.get('in_stock_only'):
# All cases
changed_detected = True
# Always record the new checksum
update_obj["previous_md5"] = fetched_md5
return changed_detected, update_obj, self.fetcher.instock_data.encode('utf-8').strip()