From 3705ce668156a5b206a46e0b24c789ca6e7bd483 Mon Sep 17 00:00:00 2001 From: dgtlmoon Date: Sat, 24 Sep 2022 23:48:03 +0200 Subject: [PATCH 01/12] Render Extract Configurable Delay Seconds should also apply after executing any JS #958 --- changedetectionio/content_fetcher.py | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/changedetectionio/content_fetcher.py b/changedetectionio/content_fetcher.py index f3af6cb1..c54f0a14 100644 --- a/changedetectionio/content_fetcher.py +++ b/changedetectionio/content_fetcher.py @@ -402,6 +402,11 @@ class base_html_playwright(Fetcher): raise JSActionExceptions(status_code=response.status, screenshot=error_screenshot, message=str(e), url=url) + else: + # JS eval was run, now we also wait some time if possible to let the page settle + if self.render_extract_delay: + page.wait_for_timeout(self.render_extract_delay * 1000) + page.wait_for_timeout(500) self.content = page.content() From ac98bc9144fac35d894739a782b350ff508af260 Mon Sep 17 00:00:00 2001 From: dgtlmoon Date: Sat, 24 Sep 2022 23:51:26 +0200 Subject: [PATCH 02/12] Upgrade Playwright to 1.26 --- Dockerfile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Dockerfile b/Dockerfile index 4e797e5c..8e528ace 100644 --- a/Dockerfile +++ b/Dockerfile @@ -22,7 +22,7 @@ RUN pip install --target=/dependencies -r /requirements.txt # Playwright is an alternative to Selenium # Excluded this package from requirements.txt to prevent arm/v6 and arm/v7 builds from failing -RUN pip install --target=/dependencies playwright~=1.25 \ +RUN pip install --target=/dependencies playwright~=1.26 \ || echo "WARN: Failed to install Playwright. The application can still run, but the Playwright option will be disabled." # Final image stage From 3ebb2ab9ba593bea346c8ca20364f8690568170b Mon Sep 17 00:00:00 2001 From: dgtlmoon Date: Sun, 25 Sep 2022 11:05:07 +0200 Subject: [PATCH 03/12] Selenium fetcher - screenshot should be taken after 'wait' time, not before #873 --- changedetectionio/content_fetcher.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/changedetectionio/content_fetcher.py b/changedetectionio/content_fetcher.py index c54f0a14..6742f01c 100644 --- a/changedetectionio/content_fetcher.py +++ b/changedetectionio/content_fetcher.py @@ -525,8 +525,6 @@ class base_html_webdriver(Fetcher): # Selenium doesn't automatically wait for actions as good as Playwright, so wait again self.driver.implicitly_wait(int(os.getenv("WEBDRIVER_DELAY_BEFORE_CONTENT_READY", 5))) - self.screenshot = self.driver.get_screenshot_as_png() - # @todo - how to check this? is it possible? self.status_code = 200 # @todo somehow we should try to get this working for WebDriver @@ -537,6 +535,8 @@ class base_html_webdriver(Fetcher): self.content = self.driver.page_source self.headers = {} + self.screenshot = self.driver.get_screenshot_as_png() + # Does the connection to the webdriver work? run a test connection. def is_ready(self): from selenium import webdriver From 738fcfe01c3e44f5860e6087e2075da4eb0bbbd5 Mon Sep 17 00:00:00 2001 From: Hmmbob <33529490+hmmbob@users.noreply.github.com> Date: Sun, 9 Oct 2022 11:42:51 +0200 Subject: [PATCH 04/12] Notification library: Bump apprise to 1.1.0 (signal, opsgenie, pagerduty, bark and mailto fixes, adds support for BulkSMS and SMSEagle) (#1002) --- requirements.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/requirements.txt b/requirements.txt index 8aaef292..15771dbc 100644 --- a/requirements.txt +++ b/requirements.txt @@ -18,7 +18,7 @@ wtforms ~= 3.0 jsonpath-ng ~= 1.5.3 # Notification library -apprise ~= 1.0.0 +apprise ~= 1.1.0 # apprise mqtt https://github.com/dgtlmoon/changedetection.io/issues/315 paho-mqtt From 71bc2fed82f51235aa63afe8713ced3c7151661c Mon Sep 17 00:00:00 2001 From: dgtlmoon Date: Sun, 9 Oct 2022 14:06:07 +0200 Subject: [PATCH 05/12] Remove quotationspage default watch --- changedetectionio/store.py | 2 -- 1 file changed, 2 deletions(-) diff --git a/changedetectionio/store.py b/changedetectionio/store.py index 4eb5dcd0..a0326e41 100644 --- a/changedetectionio/store.py +++ b/changedetectionio/store.py @@ -81,8 +81,6 @@ class ChangeDetectionStore: except (FileNotFoundError, json.decoder.JSONDecodeError): if include_default_watches: print("Creating JSON store at", self.datastore_path) - - self.add_watch(url='http://www.quotationspage.com/random.php', tag='test') self.add_watch(url='https://news.ycombinator.com/', tag='Tech news') self.add_watch(url='https://changedetection.io/CHANGELOG.txt', tag='changedetection.io') From cd467df97ad0adbb5b2cd7192891e3cdf7954c44 Mon Sep 17 00:00:00 2001 From: dgtlmoon Date: Sun, 9 Oct 2022 15:51:57 +0200 Subject: [PATCH 06/12] Adding link to BrightData Proxy info (#1003) --- changedetectionio/templates/edit.html | 1 + changedetectionio/templates/settings.html | 2 ++ docs/proxy-example.jpg | Bin 0 -> 46978 bytes 3 files changed, 3 insertions(+) create mode 100644 docs/proxy-example.jpg diff --git a/changedetectionio/templates/edit.html b/changedetectionio/templates/edit.html index 231c2016..64e9cee3 100644 --- a/changedetectionio/templates/edit.html +++ b/changedetectionio/templates/edit.html @@ -77,6 +77,7 @@

Use the Basic method (default) where your watched site doesn't need Javascript to render.

The Chrome/Javascript method requires a network connection to a running WebDriver+Chrome server, set by the ENV var 'WEBDRIVER_URL'.

+ Tip: Connect using BrightData Proxies, find out more here.
{% if form.proxy %} diff --git a/changedetectionio/templates/settings.html b/changedetectionio/templates/settings.html index 2db8c8b6..c912105e 100644 --- a/changedetectionio/templates/settings.html +++ b/changedetectionio/templates/settings.html @@ -99,6 +99,8 @@

Use the Basic method (default) where your watched sites don't need Javascript to render.

The Chrome/Javascript method requires a network connection to a running WebDriver+Chrome server, set by the ENV var 'WEBDRIVER_URL'.

+
+ Tip: Connect using BrightData Proxies, find out more here.
diff --git a/docs/proxy-example.jpg b/docs/proxy-example.jpg new file mode 100644 index 0000000000000000000000000000000000000000..60b391b4ac877e729e40341dc36a3b0aa4caedc2 GIT binary patch literal 46978 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 V4iqZN)hKS9FVi}ei1@eqKLDX!*~$O_ literal 0 HcmV?d00001 From dfa7fc3a81aab616d095ff6716778f6ad219bc1f Mon Sep 17 00:00:00 2001 From: Yusef Ouda Date: Sun, 9 Oct 2022 09:12:45 -0500 Subject: [PATCH 07/12] Adds support for jq JSON path querying engine (#1001) --- README-pip.md | 2 +- README.md | 53 ++++++++++++++-- changedetectionio/fetch_site_status.py | 5 +- changedetectionio/forms.py | 15 +++++ changedetectionio/html_tools.py | 30 +++++---- changedetectionio/templates/edit.html | 10 ++- ...lector.py => test_jsonpath_jq_selector.py} | 61 ++++++++++++++----- requirements.txt | 1 + 8 files changed, 141 insertions(+), 36 deletions(-) rename changedetectionio/tests/{test_jsonpath_selector.py => test_jsonpath_jq_selector.py} (84%) diff --git a/README-pip.md b/README-pip.md index 746175db..b6a00d32 100644 --- a/README-pip.md +++ b/README-pip.md @@ -33,7 +33,7 @@ _Need an actual Chrome runner with Javascript support? We support fetching via W #### Key Features - Lots of trigger filters, such as "Trigger on text", "Remove text by selector", "Ignore text", "Extract text", also using regular-expressions! -- Target elements with xPath and CSS Selectors, Easily monitor complex JSON with JsonPath rules +- Target elements with xPath and CSS Selectors, Easily monitor complex JSON with JSONPath or jq - Switch between fast non-JS and Chrome JS based "fetchers" - Easily specify how often a site should be checked - Execute JS before extracting text (Good for logging in, see examples in the UI!) diff --git a/README.md b/README.md index 0d08d129..797f8c56 100644 --- a/README.md +++ b/README.md @@ -47,7 +47,7 @@ _Need an actual Chrome runner with Javascript support? We support fetching via W #### Key Features - Lots of trigger filters, such as "Trigger on text", "Remove text by selector", "Ignore text", "Extract text", also using regular-expressions! -- Target elements with xPath and CSS Selectors, Easily monitor complex JSON with JsonPath rules +- Target elements with xPath and CSS Selectors, Easily monitor complex JSON with JSONPath or jq - Switch between fast non-JS and Chrome JS based "fetchers" - Easily specify how often a site should be checked - Execute JS before extracting text (Good for logging in, see examples in the UI!) @@ -121,7 +121,7 @@ See the wiki for more information https://github.com/dgtlmoon/changedetection.io ## Filters -XPath, JSONPath and CSS support comes baked in! You can be as specific as you need, use XPath exported from various XPath element query creation tools. +XPath, JSONPath, jq, and CSS support comes baked in! You can be as specific as you need, use XPath exported from various XPath element query creation tools. (We support LXML `re:test`, `re:math` and `re:replace`.) @@ -151,7 +151,7 @@ Now you can also customise your notification content! ## JSON API Monitoring -Detect changes and monitor data in JSON API's by using the built-in JSONPath selectors as a filter / selector. +Detect changes and monitor data in JSON API's by using either JSONPath or jq to filter, parse, and restructure JSON as needed. ![image](https://raw.githubusercontent.com/dgtlmoon/changedetection.io/master/docs/json-filter-field-example.png) @@ -159,9 +159,52 @@ This will re-parse the JSON and apply formatting to the text, making it super ea ![image](https://raw.githubusercontent.com/dgtlmoon/changedetection.io/master/docs/json-diff-example.png) +### JSONPath or jq? + +For more complex parsing, filtering, and modifying of JSON data, jq is recommended due to the built-in operators and functions. Refer to the [documentation](https://stedolan.github.io/jq/manual/) for more information on jq. + +The example below adds the price in dollars to each item in the JSON data, and then filters to only show items that are greater than 10. + +#### Sample input data from API +``` +{ + "items": [ + { + "name": "Product A", + "priceInCents": 2500 + }, + { + "name": "Product B", + "priceInCents": 500 + }, + { + "name": "Product C", + "priceInCents": 2000 + } + ] +} +``` + +#### Sample jq +`jq:.items[] | . + { "priceInDollars": (.priceInCents / 100) } | select(.priceInDollars > 10)` + +#### Sample output data +``` +{ + "name": "Product A", + "priceInCents": 2500, + "priceInDollars": 25 +} +{ + "name": "Product C", + "priceInCents": 2000, + "priceInDollars": 20 +} +``` + ### Parse JSON embedded in HTML! -When you enable a `json:` filter, you can even automatically extract and parse embedded JSON inside a HTML page! Amazingly handy for sites that build content based on JSON, such as many e-commerce websites. +When you enable a `json:` or `jq:` filter, you can even automatically extract and parse embedded JSON inside a HTML page! Amazingly handy for sites that build content based on JSON, such as many e-commerce websites. ``` @@ -171,7 +214,7 @@ When you enable a `json:` filter, you can even automatically extract and parse e ``` -`json:$.price` would give `23.50`, or you can extract the whole structure +`json:$.price` or `jq:.price` would give `23.50`, or you can extract the whole structure ## Proxy configuration diff --git a/changedetectionio/fetch_site_status.py b/changedetectionio/fetch_site_status.py index 79e282b5..0f84da16 100644 --- a/changedetectionio/fetch_site_status.py +++ b/changedetectionio/fetch_site_status.py @@ -141,8 +141,9 @@ class perform_site_check(): has_filter_rule = True if has_filter_rule: - if 'json:' in css_filter_rule: - stripped_text_from_html = html_tools.extract_json_as_string(content=fetcher.content, jsonpath_filter=css_filter_rule) + json_filter_prefixes = ['json:', 'jq:'] + if any(prefix in css_filter_rule for prefix in json_filter_prefixes): + stripped_text_from_html = html_tools.extract_json_as_string(content=fetcher.content, json_filter=css_filter_rule) is_html = False if is_html or is_source: diff --git a/changedetectionio/forms.py b/changedetectionio/forms.py index 279f7c7f..7fa17f90 100644 --- a/changedetectionio/forms.py +++ b/changedetectionio/forms.py @@ -304,6 +304,21 @@ class ValidateCSSJSONXPATHInput(object): # Re #265 - maybe in the future fetch the page and offer a # warning/notice that its possible the rule doesnt yet match anything? + if 'jq:' in line: + if not self.allow_json: + raise ValidationError("jq not permitted in this field!") + + import jq + input = line.replace('jq:', '') + + try: + jq.compile(input) + except (ValueError) as e: + message = field.gettext('\'%s\' is not a valid jq expression. (%s)') + raise ValidationError(message % (input, str(e))) + except: + raise ValidationError("A system-error occurred when validating your jq expression") + class quickWatchForm(Form): url = fields.URLField('URL', validators=[validateURL()]) diff --git a/changedetectionio/html_tools.py b/changedetectionio/html_tools.py index a851a4d6..6cc8e20a 100644 --- a/changedetectionio/html_tools.py +++ b/changedetectionio/html_tools.py @@ -3,6 +3,7 @@ from typing import List from bs4 import BeautifulSoup from jsonpath_ng.ext import parse +import jq import re from inscriptis import get_text from inscriptis.model.config import ParserConfig @@ -79,19 +80,26 @@ def extract_element(find='title', html_content=''): return element_text # -def _parse_json(json_data, jsonpath_filter): - s=[] - jsonpath_expression = parse(jsonpath_filter.replace('json:', '')) - match = jsonpath_expression.find(json_data) - +def _parse_json(json_data, json_filter): + if 'json:' in json_filter: + jsonpath_expression = parse(json_filter.replace('json:', '')) + match = jsonpath_expression.find(json_data) + return _get_stripped_text_from_json_match(match) + if 'jq:' in json_filter: + jq_expression = jq.compile(json_filter.replace('jq:', '')) + match = jq_expression.input(json_data).all() + return _get_stripped_text_from_json_match(match) + +def _get_stripped_text_from_json_match(match): + s = [] # More than one result, we will return it as a JSON list. if len(match) > 1: for i in match: - s.append(i.value) + s.append(i.value if hasattr(i, 'value') else i) # Single value, use just the value, as it could be later used in a token in notifications. if len(match) == 1: - s = match[0].value + s = match[0].value if hasattr(match[0], 'value') else match[0] # Re #257 - Better handling where it does not exist, in the case the original 's' value was False.. if not match: @@ -103,16 +111,16 @@ def _parse_json(json_data, jsonpath_filter): return stripped_text_from_html -def extract_json_as_string(content, jsonpath_filter): +def extract_json_as_string(content, json_filter): stripped_text_from_html = False # Try to parse/filter out the JSON, if we get some parser error, then maybe it's embedded blob.. just return the first that matches jsonpath_filter + # Foreach blob.. just return the first that matches json_filter s = [] soup = BeautifulSoup(content, 'html.parser') bs_result = soup.findAll('script') @@ -131,7 +139,7 @@ def extract_json_as_string(content, jsonpath_filter): # Just skip it continue else: - stripped_text_from_html = _parse_json(json_data, jsonpath_filter) + stripped_text_from_html = _parse_json(json_data, json_filter) if stripped_text_from_html: break diff --git a/changedetectionio/templates/edit.html b/changedetectionio/templates/edit.html index 64e9cee3..907894e1 100644 --- a/changedetectionio/templates/edit.html +++ b/changedetectionio/templates/edit.html @@ -184,8 +184,12 @@ User-Agent: wonderbra 1.0") }}

diff --git a/changedetectionio/tests/test_jsonpath_selector.py b/changedetectionio/tests/test_jsonpath_jq_selector.py similarity index 84% rename from changedetectionio/tests/test_jsonpath_selector.py rename to changedetectionio/tests/test_jsonpath_jq_selector.py index 729a201d..d0082122 100644 --- a/changedetectionio/tests/test_jsonpath_selector.py +++ b/changedetectionio/tests/test_jsonpath_jq_selector.py @@ -2,7 +2,7 @@ # coding=utf-8 import time -from flask import url_for +from flask import url_for, escape from . util import live_server_setup import pytest @@ -36,16 +36,26 @@ and it can also be repeated from .. import html_tools # See that we can find the second