Compare commits
5 Commits
4a1f6d052c
...
f7afcfe991
Author | SHA1 | Date |
---|---|---|
Massaki Archambault | f7afcfe991 | |
Massaki Archambault | b38df4b298 | |
Massaki Archambault | 0e22353da8 | |
Massaki Archambault | 51968e443a | |
Massaki Archambault | bda9b43318 |
|
@ -202,4 +202,4 @@ tags
|
||||||
[._]*.un~
|
[._]*.un~
|
||||||
|
|
||||||
### Project-specific
|
### Project-specific
|
||||||
dealwatch.yml
|
ecommerce-exporter.yml
|
|
@ -2,4 +2,4 @@ FROM python:3.10
|
||||||
COPY . /tmp/package
|
COPY . /tmp/package
|
||||||
RUN pip install --no-cache-dir /tmp/package && \
|
RUN pip install --no-cache-dir /tmp/package && \
|
||||||
rm -r /tmp/package
|
rm -r /tmp/package
|
||||||
ENTRYPOINT ["dealwatch"]
|
ENTRYPOINT ["ecommerce-exporter"]
|
|
@ -1,36 +0,0 @@
|
||||||
import re
|
|
||||||
|
|
||||||
import httpx
|
|
||||||
import parsel
|
|
||||||
|
|
||||||
class ScrapeTarget:
|
|
||||||
def __init__(self, product_name, target_name, url, selector, regex=None):
|
|
||||||
self.product_name = product_name
|
|
||||||
self.target_name = target_name
|
|
||||||
self.url = url
|
|
||||||
self.selector = selector+'::text'
|
|
||||||
self.regex = re.compile(regex if regex else r'[0-9]+(\.[0-9]{2})?')
|
|
||||||
self.headers = {}
|
|
||||||
|
|
||||||
def query_target(self):
|
|
||||||
print('Query product %s, target %s' % (self.product_name, self.target_name))
|
|
||||||
# some sites get suspicious if we talk to them in HTTP/1.1
|
|
||||||
# we use httpx to have HTTP2 support and circumvent that issue
|
|
||||||
query_response = httpx.get(
|
|
||||||
url=self.url,
|
|
||||||
headers=self.headers,
|
|
||||||
follow_redirects=True,
|
|
||||||
).text
|
|
||||||
selector = parsel.Selector(text=query_response)
|
|
||||||
|
|
||||||
# Match the selector
|
|
||||||
selector_match = selector.css(self.selector).get()
|
|
||||||
if selector_match:
|
|
||||||
# Match the regex
|
|
||||||
regex_match = self.regex.search(selector_match)
|
|
||||||
if regex_match:
|
|
||||||
str_result = regex_match.group(0)
|
|
||||||
# Convert the reult to float
|
|
||||||
float_result = float(str_result)
|
|
||||||
return float_result
|
|
||||||
return None
|
|
|
@ -1,10 +1,29 @@
|
||||||
import argparse
|
import argparse
|
||||||
|
import os
|
||||||
|
import time
|
||||||
|
|
||||||
import yaml
|
import yaml
|
||||||
|
|
||||||
from prometheus_client import start_http_server
|
from httpx import RequestError
|
||||||
|
from prometheus_client import start_http_server, Gauge, Counter
|
||||||
|
|
||||||
from dealwatch.scrape_target import ScrapeTarget
|
from ecommerce_exporter.scrape_target import ScrapeError, ScrapeTarget
|
||||||
|
|
||||||
|
ECOMMERCE_SCRAPE_TARGET_VALUE = Gauge(
|
||||||
|
'ecommerce_scrape_target_value',
|
||||||
|
'The value scraped from a scrape target',
|
||||||
|
['product_name', 'target_name'],
|
||||||
|
)
|
||||||
|
ECOMMERCE_SCRAPE_TARGET_SUCCESS = Counter(
|
||||||
|
'ecommerce_scrape_target_success_total',
|
||||||
|
'The number of successful scrape and parse of a scrape target',
|
||||||
|
['product_name', 'target_name'],
|
||||||
|
)
|
||||||
|
ECOMMERCE_SCRAPE_TARGET_FAILURE = Counter(
|
||||||
|
'ecommerce_scrape_target_failure_total',
|
||||||
|
'The number of failed scrape and parse of a scrape target',
|
||||||
|
['product_name', 'target_name', 'exception'],
|
||||||
|
)
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
parser = argparse.ArgumentParser("An utility to scrape e-commerce product price and expose them as prometheus metrics")
|
parser = argparse.ArgumentParser("An utility to scrape e-commerce product price and expose them as prometheus metrics")
|
||||||
|
@ -12,7 +31,13 @@ def main():
|
||||||
'-c', '--config',
|
'-c', '--config',
|
||||||
help='The configuration file. (default: %(default)s)',
|
help='The configuration file. (default: %(default)s)',
|
||||||
type=str,
|
type=str,
|
||||||
default='dealwatch.yml',
|
default='ecommerce-exporter.yml',
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'-i', '--interval',
|
||||||
|
help='The target scrape interval, in minutes. (default: %(default)s)',
|
||||||
|
type=float,
|
||||||
|
default=15,
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
'--user-agent',
|
'--user-agent',
|
||||||
|
@ -34,7 +59,7 @@ def main():
|
||||||
)
|
)
|
||||||
|
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
scrape_targets = parse_config(args.config)
|
scrape_targets = parse_config(os.path.abspath(args.config))
|
||||||
|
|
||||||
# setup the headers for each scrape targets
|
# setup the headers for each scrape targets
|
||||||
for scrape_target in scrape_targets:
|
for scrape_target in scrape_targets:
|
||||||
|
@ -46,8 +71,28 @@ def main():
|
||||||
# start the http server to server the prometheus metrics
|
# start the http server to server the prometheus metrics
|
||||||
start_http_server(args.listen_port, args.listen_address)
|
start_http_server(args.listen_port, args.listen_address)
|
||||||
|
|
||||||
|
# start the main loop
|
||||||
|
while True:
|
||||||
for scrape_target in scrape_targets:
|
for scrape_target in scrape_targets:
|
||||||
print(scrape_target.query_target())
|
try:
|
||||||
|
print("Starting scrape. product: '%s', target '%s'" % (scrape_target.product_name, scrape_target.target_name))
|
||||||
|
value = scrape_target.query_target()
|
||||||
|
ECOMMERCE_SCRAPE_TARGET_VALUE.labels(
|
||||||
|
product_name=scrape_target.product_name,
|
||||||
|
target_name=scrape_target.target_name
|
||||||
|
).set(value)
|
||||||
|
ECOMMERCE_SCRAPE_TARGET_SUCCESS.labels(
|
||||||
|
product_name=scrape_target.product_name,
|
||||||
|
target_name=scrape_target.target_name,
|
||||||
|
).inc()
|
||||||
|
except (RequestError, ScrapeError) as e:
|
||||||
|
print("Failed to scrape! product: '%s', target: '%s', message: '%s'" % (scrape_target.product_name, scrape_target.target_name, e))
|
||||||
|
ECOMMERCE_SCRAPE_TARGET_FAILURE.labels(
|
||||||
|
product_name=scrape_target.product_name,
|
||||||
|
target_name=scrape_target.target_name,
|
||||||
|
exception=e.__class__.__name__,
|
||||||
|
).inc()
|
||||||
|
time.sleep(args.interval * 60)
|
||||||
|
|
||||||
def parse_config(config_filename):
|
def parse_config(config_filename):
|
||||||
result = []
|
result = []
|
||||||
|
@ -66,10 +111,11 @@ def parse_config(config_filename):
|
||||||
# Create a ScrapeTarget for each targets to scrape
|
# Create a ScrapeTarget for each targets to scrape
|
||||||
result.append(ScrapeTarget(
|
result.append(ScrapeTarget(
|
||||||
product_name=product_name,
|
product_name=product_name,
|
||||||
target_name=get_field_or_die(target, 'name'),
|
|
||||||
url=get_field_or_die(target, 'url'),
|
url=get_field_or_die(target, 'url'),
|
||||||
selector=get_field_or_die(target, 'selector'),
|
selector=get_field_or_die(target, 'selector'),
|
||||||
|
target_name=target.get('name'),
|
||||||
regex=target.get('regex'),
|
regex=target.get('regex'),
|
||||||
|
parser=target.get('parser'),
|
||||||
))
|
))
|
||||||
return result
|
return result
|
||||||
|
|
|
@ -0,0 +1,62 @@
|
||||||
|
import json
|
||||||
|
import re
|
||||||
|
|
||||||
|
from urllib.parse import urlparse
|
||||||
|
|
||||||
|
import httpx
|
||||||
|
import parsel
|
||||||
|
import pyjq
|
||||||
|
|
||||||
|
class ScrapeTarget:
|
||||||
|
def __init__(self, product_name, url, selector, target_name=None, regex=None, parser=None):
|
||||||
|
self.product_name = product_name
|
||||||
|
self.target_name = target_name if target_name else urlparse(url).hostname
|
||||||
|
self.url = url
|
||||||
|
self.selector = selector
|
||||||
|
self.regex = re.compile(regex if regex else r'[0-9]+(\.[0-9]{2})?')
|
||||||
|
self.parser = parser if parser else 'html'
|
||||||
|
self.headers = {}
|
||||||
|
|
||||||
|
# sanity check
|
||||||
|
valid_parsers = ('html', 'json')
|
||||||
|
if self.parser not in valid_parsers:
|
||||||
|
raise ValueError("Invalid parser configured (got '%s' but need one of %s) product: '%s', target: '%s'" % (self.parser, valid_parsers, self.product_name, self.target_name))
|
||||||
|
|
||||||
|
def query_target(self):
|
||||||
|
# some sites get suspicious if we talk to them in HTTP/1.1 (maybe because it doesn't match our user-agent?)
|
||||||
|
# we use httpx to have HTTP2 support and circumvent that issue
|
||||||
|
query_response = httpx.get(
|
||||||
|
url=self.url,
|
||||||
|
headers=self.headers,
|
||||||
|
follow_redirects=True,
|
||||||
|
).text
|
||||||
|
|
||||||
|
# parse the response and match the selector
|
||||||
|
selector_match = ''
|
||||||
|
if self.parser == 'html':
|
||||||
|
# parse response as html
|
||||||
|
selector = parsel.Selector(text=query_response)
|
||||||
|
selector_match = selector.css(self.selector).get()
|
||||||
|
elif self.parser == 'json':
|
||||||
|
# parse response as json
|
||||||
|
query_response_json = json.loads(query_response)
|
||||||
|
selector_match = str(pyjq.first(self.selector, query_response_json))
|
||||||
|
else:
|
||||||
|
raise ScrapeError('Invalid parser!')
|
||||||
|
|
||||||
|
if not selector_match:
|
||||||
|
raise ScrapeError('Failed to match selector!')
|
||||||
|
|
||||||
|
# match the regex
|
||||||
|
regex_match = self.regex.search(selector_match)
|
||||||
|
if regex_match:
|
||||||
|
str_result = regex_match.group(0)
|
||||||
|
# convert the result to float
|
||||||
|
float_result = float(str_result)
|
||||||
|
return float_result
|
||||||
|
else:
|
||||||
|
raise ScrapeError('Failed to match regex!')
|
||||||
|
|
||||||
|
class ScrapeError(Exception):
|
||||||
|
def __init__(self, msg):
|
||||||
|
super().__init__(msg)
|
|
@ -1,5 +1,5 @@
|
||||||
[metadata]
|
[metadata]
|
||||||
name = dealwatch
|
name = ecommerce-exporter
|
||||||
author = badjware
|
author = badjware
|
||||||
author_email = marchambault.badjware.dev
|
author_email = marchambault.badjware.dev
|
||||||
platform = any
|
platform = any
|
||||||
|
@ -13,10 +13,11 @@ install_requires=
|
||||||
PyYAML~=6.0
|
PyYAML~=6.0
|
||||||
httpx~=0.23.0
|
httpx~=0.23.0
|
||||||
parsel~=1.6.0
|
parsel~=1.6.0
|
||||||
|
pyjq~=2.6.0
|
||||||
prometheus-client~=0.15.0
|
prometheus-client~=0.15.0
|
||||||
|
|
||||||
[options.entry_points]
|
[options.entry_points]
|
||||||
console_scripts =
|
console_scripts =
|
||||||
dealwatch = dealwatch.cli:main
|
ecommerce-exporter = ecommerce_exporter.cli:main
|
||||||
|
|
||||||
[tool.setuptools_scm]
|
[tool.setuptools_scm]
|
Loading…
Reference in New Issue