Compare commits
40 Commits
store-watc
...
2727-notif
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cd919dbb0b | ||
|
|
6e0a0b5d5f | ||
|
|
a1d8186546 | ||
|
|
d82f254382 | ||
|
|
63366a0a83 | ||
|
|
eba27dcd91 | ||
|
|
c1dd681643 | ||
|
|
ecafa27833 | ||
|
|
f7d4e58613 | ||
|
|
5bb47e47db | ||
|
|
03151da68e | ||
|
|
a16a70229d | ||
|
|
9476c1076b | ||
|
|
a4959b5971 | ||
|
|
a278fa22f2 | ||
|
|
d39530b261 | ||
|
|
d4b4355ff5 | ||
|
|
c1c8de3104 | ||
|
|
5a768d7db3 | ||
|
|
f38429ec93 | ||
|
|
783926962d | ||
|
|
6cd1d50a4f | ||
|
|
54a4970a4c | ||
|
|
fd00453e6d | ||
|
|
2842ffb205 | ||
|
|
ec4e2f5649 | ||
|
|
fe8e3d1cb1 | ||
|
|
69fbafbdb7 | ||
|
|
f255165571 | ||
|
|
7ff34baa90 | ||
|
|
043378d09c | ||
|
|
af4bafcff8 | ||
|
|
b656338c63 | ||
|
|
97af190910 | ||
|
|
e9e063e18e | ||
|
|
45c444d0db | ||
|
|
00458b95c4 | ||
|
|
dad9760832 | ||
|
|
e2c2a76cb2 | ||
|
|
5b34aece96 |
@@ -37,6 +37,7 @@ RUN pip install --target=/dependencies playwright~=1.41.2 \
|
|||||||
|
|
||||||
# Final image stage
|
# Final image stage
|
||||||
FROM python:${PYTHON_VERSION}-slim-bookworm
|
FROM python:${PYTHON_VERSION}-slim-bookworm
|
||||||
|
LABEL org.opencontainers.image.source="https://github.com/dgtlmoon/changedetection.io"
|
||||||
|
|
||||||
RUN apt-get update && apt-get install -y --no-install-recommends \
|
RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||||
libxslt1.1 \
|
libxslt1.1 \
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
recursive-include changedetectionio/api *
|
recursive-include changedetectionio/api *
|
||||||
|
recursive-include changedetectionio/apprise_plugin *
|
||||||
recursive-include changedetectionio/blueprint *
|
recursive-include changedetectionio/blueprint *
|
||||||
recursive-include changedetectionio/content_fetchers *
|
recursive-include changedetectionio/content_fetchers *
|
||||||
recursive-include changedetectionio/model *
|
recursive-include changedetectionio/model *
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
# Read more https://github.com/dgtlmoon/changedetection.io/wiki
|
# Read more https://github.com/dgtlmoon/changedetection.io/wiki
|
||||||
|
|
||||||
__version__ = '0.46.04'
|
__version__ = '0.47.03'
|
||||||
|
|
||||||
from changedetectionio.strtobool import strtobool
|
from changedetectionio.strtobool import strtobool
|
||||||
from json.decoder import JSONDecodeError
|
from json.decoder import JSONDecodeError
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ class Watch(Resource):
|
|||||||
abort(404, message='No watch exists with the UUID of {}'.format(uuid))
|
abort(404, message='No watch exists with the UUID of {}'.format(uuid))
|
||||||
|
|
||||||
if request.args.get('recheck'):
|
if request.args.get('recheck'):
|
||||||
self.update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': True}))
|
self.update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
return "OK", 200
|
return "OK", 200
|
||||||
if request.args.get('paused', '') == 'paused':
|
if request.args.get('paused', '') == 'paused':
|
||||||
self.datastore.data['watching'].get(uuid).pause()
|
self.datastore.data['watching'].get(uuid).pause()
|
||||||
@@ -246,7 +246,7 @@ class CreateWatch(Resource):
|
|||||||
|
|
||||||
new_uuid = self.datastore.add_watch(url=url, extras=extras, tag=tags)
|
new_uuid = self.datastore.add_watch(url=url, extras=extras, tag=tags)
|
||||||
if new_uuid:
|
if new_uuid:
|
||||||
self.update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': new_uuid, 'skip_when_checksum_same': True}))
|
self.update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': new_uuid}))
|
||||||
return {'uuid': new_uuid}, 201
|
return {'uuid': new_uuid}, 201
|
||||||
else:
|
else:
|
||||||
return "Invalid or unsupported URL", 400
|
return "Invalid or unsupported URL", 400
|
||||||
@@ -303,7 +303,7 @@ class CreateWatch(Resource):
|
|||||||
|
|
||||||
if request.args.get('recheck_all'):
|
if request.args.get('recheck_all'):
|
||||||
for uuid in self.datastore.data['watching'].keys():
|
for uuid in self.datastore.data['watching'].keys():
|
||||||
self.update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': True}))
|
self.update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
return {'status': "OK"}, 200
|
return {'status': "OK"}, 200
|
||||||
|
|
||||||
return list, 200
|
return list, 200
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
# include the decorator
|
# include the decorator
|
||||||
from apprise.decorators import notify
|
from apprise.decorators import notify
|
||||||
|
from loguru import logger
|
||||||
|
|
||||||
@notify(on="delete")
|
@notify(on="delete")
|
||||||
@notify(on="deletes")
|
@notify(on="deletes")
|
||||||
@@ -64,10 +65,12 @@ def apprise_custom_api_call_wrapper(body, title, notify_type, *args, **kwargs):
|
|||||||
auth = (URLBase.unquote(results.get('user')))
|
auth = (URLBase.unquote(results.get('user')))
|
||||||
|
|
||||||
# Try to auto-guess if it's JSON
|
# Try to auto-guess if it's JSON
|
||||||
|
h = 'application/json; charset=utf-8'
|
||||||
try:
|
try:
|
||||||
json.loads(body)
|
json.loads(body)
|
||||||
headers['Content-Type'] = 'application/json; charset=utf-8'
|
headers['Content-Type'] = h
|
||||||
except ValueError as e:
|
except ValueError as e:
|
||||||
|
logger.warning(f"Could not automatically add '{h}' header to the {kwargs['meta'].get('schema')}:// notification because the document failed to parse as JSON: {e}")
|
||||||
pass
|
pass
|
||||||
|
|
||||||
r(results.get('url'),
|
r(results.get('url'),
|
||||||
|
|||||||
@@ -1,4 +1,7 @@
|
|||||||
|
import importlib
|
||||||
from concurrent.futures import ThreadPoolExecutor
|
from concurrent.futures import ThreadPoolExecutor
|
||||||
|
|
||||||
|
from changedetectionio.processors.text_json_diff.processor import FilterNotFoundInResponse
|
||||||
from changedetectionio.store import ChangeDetectionStore
|
from changedetectionio.store import ChangeDetectionStore
|
||||||
|
|
||||||
from functools import wraps
|
from functools import wraps
|
||||||
@@ -30,7 +33,6 @@ def construct_blueprint(datastore: ChangeDetectionStore):
|
|||||||
def long_task(uuid, preferred_proxy):
|
def long_task(uuid, preferred_proxy):
|
||||||
import time
|
import time
|
||||||
from changedetectionio.content_fetchers import exceptions as content_fetcher_exceptions
|
from changedetectionio.content_fetchers import exceptions as content_fetcher_exceptions
|
||||||
from changedetectionio.processors.text_json_diff import text_json_diff
|
|
||||||
from changedetectionio.safe_jinja import render as jinja_render
|
from changedetectionio.safe_jinja import render as jinja_render
|
||||||
|
|
||||||
status = {'status': '', 'length': 0, 'text': ''}
|
status = {'status': '', 'length': 0, 'text': ''}
|
||||||
@@ -38,8 +40,12 @@ def construct_blueprint(datastore: ChangeDetectionStore):
|
|||||||
contents = ''
|
contents = ''
|
||||||
now = time.time()
|
now = time.time()
|
||||||
try:
|
try:
|
||||||
update_handler = text_json_diff.perform_site_check(datastore=datastore, watch_uuid=uuid)
|
processor_module = importlib.import_module("changedetectionio.processors.text_json_diff.processor")
|
||||||
update_handler.call_browser()
|
update_handler = processor_module.perform_site_check(datastore=datastore,
|
||||||
|
watch_uuid=uuid
|
||||||
|
)
|
||||||
|
|
||||||
|
update_handler.call_browser(preferred_proxy_id=preferred_proxy)
|
||||||
# title, size is len contents not len xfer
|
# title, size is len contents not len xfer
|
||||||
except content_fetcher_exceptions.Non200ErrorCodeReceived as e:
|
except content_fetcher_exceptions.Non200ErrorCodeReceived as e:
|
||||||
if e.status_code == 404:
|
if e.status_code == 404:
|
||||||
@@ -48,7 +54,7 @@ def construct_blueprint(datastore: ChangeDetectionStore):
|
|||||||
status.update({'status': 'ERROR', 'length': len(contents), 'text': f"{e.status_code} - Access denied"})
|
status.update({'status': 'ERROR', 'length': len(contents), 'text': f"{e.status_code} - Access denied"})
|
||||||
else:
|
else:
|
||||||
status.update({'status': 'ERROR', 'length': len(contents), 'text': f"Status code: {e.status_code}"})
|
status.update({'status': 'ERROR', 'length': len(contents), 'text': f"Status code: {e.status_code}"})
|
||||||
except text_json_diff.FilterNotFoundInResponse:
|
except FilterNotFoundInResponse:
|
||||||
status.update({'status': 'OK', 'length': len(contents), 'text': f"OK but CSS/xPath filter not found (page changed layout?)"})
|
status.update({'status': 'OK', 'length': len(contents), 'text': f"OK but CSS/xPath filter not found (page changed layout?)"})
|
||||||
except content_fetcher_exceptions.EmptyReply as e:
|
except content_fetcher_exceptions.EmptyReply as e:
|
||||||
if e.status_code == 403 or e.status_code == 401:
|
if e.status_code == 403 or e.status_code == 401:
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ def construct_blueprint(datastore: ChangeDetectionStore, update_q: PriorityQueue
|
|||||||
datastore.data['watching'][uuid]['track_ldjson_price_data'] = PRICE_DATA_TRACK_ACCEPT
|
datastore.data['watching'][uuid]['track_ldjson_price_data'] = PRICE_DATA_TRACK_ACCEPT
|
||||||
datastore.data['watching'][uuid]['processor'] = 'restock_diff'
|
datastore.data['watching'][uuid]['processor'] = 'restock_diff'
|
||||||
datastore.data['watching'][uuid].clear_watch()
|
datastore.data['watching'][uuid].clear_watch()
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': False}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
return redirect(url_for("index"))
|
return redirect(url_for("index"))
|
||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
|
|||||||
@@ -17,7 +17,6 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<script src="{{url_for('static_content', group='js', filename='watch-settings.js')}}" defer></script>
|
<script src="{{url_for('static_content', group='js', filename='watch-settings.js')}}" defer></script>
|
||||||
<!--<script src="{{url_for('static_content', group='js', filename='limit.js')}}" defer></script>-->
|
|
||||||
<script src="{{url_for('static_content', group='js', filename='notifications.js')}}" defer></script>
|
<script src="{{url_for('static_content', group='js', filename='notifications.js')}}" defer></script>
|
||||||
|
|
||||||
<div class="edit-form monospaced-textarea">
|
<div class="edit-form monospaced-textarea">
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ from loguru import logger
|
|||||||
from changedetectionio.content_fetchers.exceptions import BrowserStepsStepException
|
from changedetectionio.content_fetchers.exceptions import BrowserStepsStepException
|
||||||
import os
|
import os
|
||||||
|
|
||||||
visualselector_xpath_selectors = 'div,span,form,table,tbody,tr,td,a,p,ul,li,h1,h2,h3,h4,header,footer,section,article,aside,details,main,nav,section,summary'
|
# Visual Selector scraper - 'Button' is there because some sites have <button>OUT OF STOCK</button>.
|
||||||
|
visualselector_xpath_selectors = 'div,span,form,table,tbody,tr,td,a,p,ul,li,h1,h2,h3,h4,header,footer,section,article,aside,details,main,nav,section,summary,button'
|
||||||
|
|
||||||
|
|
||||||
# available_fetchers() will scan this implementation looking for anything starting with html_
|
# available_fetchers() will scan this implementation looking for anything starting with html_
|
||||||
# this information is used in the form selections
|
# this information is used in the form selections
|
||||||
|
|||||||
@@ -75,6 +75,7 @@ class fetcher(Fetcher):
|
|||||||
self.headers = r.headers
|
self.headers = r.headers
|
||||||
|
|
||||||
if not r.content or not len(r.content):
|
if not r.content or not len(r.content):
|
||||||
|
logger.debug(f"Requests returned empty content for '{url}'")
|
||||||
if not empty_pages_are_a_change:
|
if not empty_pages_are_a_change:
|
||||||
raise EmptyReply(url=url, status_code=r.status_code)
|
raise EmptyReply(url=url, status_code=r.status_code)
|
||||||
else:
|
else:
|
||||||
|
|||||||
@@ -154,10 +154,14 @@ function isItemInStock() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
elementText = "";
|
elementText = "";
|
||||||
if (element.tagName.toLowerCase() === "input") {
|
try {
|
||||||
elementText = element.value.toLowerCase().trim();
|
if (element.tagName.toLowerCase() === "input") {
|
||||||
} else {
|
elementText = element.value.toLowerCase().trim();
|
||||||
elementText = getElementBaseText(element);
|
} else {
|
||||||
|
elementText = getElementBaseText(element);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.warn('stock-not-in-stock.js scraper - handling element for gettext failed', e);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (elementText.length) {
|
if (elementText.length) {
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
import datetime
|
import datetime
|
||||||
import importlib
|
|
||||||
|
|
||||||
import flask_login
|
import flask_login
|
||||||
import locale
|
import locale
|
||||||
@@ -12,9 +11,7 @@ import threading
|
|||||||
import time
|
import time
|
||||||
import timeago
|
import timeago
|
||||||
|
|
||||||
from .content_fetchers.exceptions import ReplyWithContentButNoText
|
|
||||||
from .processors import find_processors, get_parent_module, get_custom_watch_obj_for_processor
|
from .processors import find_processors, get_parent_module, get_custom_watch_obj_for_processor
|
||||||
from .processors.text_json_diff.processor import FilterNotFoundInResponse
|
|
||||||
from .safe_jinja import render as jinja_render
|
from .safe_jinja import render as jinja_render
|
||||||
from changedetectionio.strtobool import strtobool
|
from changedetectionio.strtobool import strtobool
|
||||||
from copy import deepcopy
|
from copy import deepcopy
|
||||||
@@ -70,7 +67,6 @@ FlaskCompress(app)
|
|||||||
|
|
||||||
# Stop browser caching of assets
|
# Stop browser caching of assets
|
||||||
app.config['SEND_FILE_MAX_AGE_DEFAULT'] = 0
|
app.config['SEND_FILE_MAX_AGE_DEFAULT'] = 0
|
||||||
|
|
||||||
app.config.exit = Event()
|
app.config.exit = Event()
|
||||||
|
|
||||||
app.config['NEW_VERSION_AVAILABLE'] = False
|
app.config['NEW_VERSION_AVAILABLE'] = False
|
||||||
@@ -473,7 +469,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
continue
|
continue
|
||||||
if watch.get('last_error'):
|
if watch.get('last_error'):
|
||||||
errored_count += 1
|
errored_count += 1
|
||||||
|
|
||||||
if search_q:
|
if search_q:
|
||||||
if (watch.get('title') and search_q in watch.get('title').lower()) or search_q in watch.get('url', '').lower():
|
if (watch.get('title') and search_q in watch.get('title').lower()) or search_q in watch.get('url', '').lower():
|
||||||
sorted_watches.append(watch)
|
sorted_watches.append(watch)
|
||||||
@@ -536,7 +532,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
@login_optionally_required
|
@login_optionally_required
|
||||||
def ajax_callback_send_notification_test(watch_uuid=None):
|
def ajax_callback_send_notification_test(watch_uuid=None):
|
||||||
|
|
||||||
# Watch_uuid could be unset in the case its used in tag editor, global setings
|
# Watch_uuid could be unset in the case it`s used in tag editor, global settings
|
||||||
import apprise
|
import apprise
|
||||||
import random
|
import random
|
||||||
from .apprise_asset import asset
|
from .apprise_asset import asset
|
||||||
@@ -545,13 +541,15 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
from changedetectionio.apprise_plugin import apprise_custom_api_call_wrapper
|
from changedetectionio.apprise_plugin import apprise_custom_api_call_wrapper
|
||||||
is_global_settings_form = request.args.get('mode', '') == 'global-settings'
|
is_global_settings_form = request.args.get('mode', '') == 'global-settings'
|
||||||
is_group_settings_form = request.args.get('mode', '') == 'group-settings'
|
is_group_settings_form = request.args.get('mode', '') == 'group-settings'
|
||||||
|
|
||||||
# Use an existing random one on the global/main settings form
|
# Use an existing random one on the global/main settings form
|
||||||
if not watch_uuid and (is_global_settings_form or is_group_settings_form):
|
if not watch_uuid and (is_global_settings_form or is_group_settings_form) \
|
||||||
|
and datastore.data.get('watching'):
|
||||||
|
|
||||||
logger.debug(f"Send test notification - Choosing random Watch {watch_uuid}")
|
logger.debug(f"Send test notification - Choosing random Watch {watch_uuid}")
|
||||||
watch_uuid = random.choice(list(datastore.data['watching'].keys()))
|
watch_uuid = random.choice(list(datastore.data['watching'].keys()))
|
||||||
|
watch = datastore.data['watching'].get(watch_uuid)
|
||||||
watch = datastore.data['watching'].get(watch_uuid)
|
else:
|
||||||
|
watch = None
|
||||||
|
|
||||||
notification_urls = request.form['notification_urls'].strip().splitlines()
|
notification_urls = request.form['notification_urls'].strip().splitlines()
|
||||||
|
|
||||||
@@ -791,7 +789,6 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
# Recast it if need be to right data Watch handler
|
# Recast it if need be to right data Watch handler
|
||||||
watch_class = get_custom_watch_obj_for_processor(form.data.get('processor'))
|
watch_class = get_custom_watch_obj_for_processor(form.data.get('processor'))
|
||||||
datastore.data['watching'][uuid] = watch_class(datastore_path=datastore_o.datastore_path, default=datastore.data['watching'][uuid])
|
datastore.data['watching'][uuid] = watch_class(datastore_path=datastore_o.datastore_path, default=datastore.data['watching'][uuid])
|
||||||
|
|
||||||
flash("Updated watch - unpaused!" if request.args.get('unpause_on_save') else "Updated watch.")
|
flash("Updated watch - unpaused!" if request.args.get('unpause_on_save') else "Updated watch.")
|
||||||
|
|
||||||
# Re #286 - We wait for syncing new data to disk in another thread every 60 seconds
|
# Re #286 - We wait for syncing new data to disk in another thread every 60 seconds
|
||||||
@@ -799,7 +796,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
datastore.needs_write_urgent = True
|
datastore.needs_write_urgent = True
|
||||||
|
|
||||||
# Queue the watch for immediate recheck, with a higher priority
|
# Queue the watch for immediate recheck, with a higher priority
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': False}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
|
|
||||||
# Diff page [edit] link should go back to diff page
|
# Diff page [edit] link should go back to diff page
|
||||||
if request.args.get("next") and request.args.get("next") == 'diff':
|
if request.args.get("next") and request.args.get("next") == 'diff':
|
||||||
@@ -980,7 +977,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
importer = import_url_list()
|
importer = import_url_list()
|
||||||
importer.run(data=request.values.get('urls'), flash=flash, datastore=datastore, processor=request.values.get('processor', 'text_json_diff'))
|
importer.run(data=request.values.get('urls'), flash=flash, datastore=datastore, processor=request.values.get('processor', 'text_json_diff'))
|
||||||
for uuid in importer.new_uuids:
|
for uuid in importer.new_uuids:
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': True}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
|
|
||||||
if len(importer.remaining_data) == 0:
|
if len(importer.remaining_data) == 0:
|
||||||
return redirect(url_for('index'))
|
return redirect(url_for('index'))
|
||||||
@@ -993,7 +990,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
d_importer = import_distill_io_json()
|
d_importer = import_distill_io_json()
|
||||||
d_importer.run(data=request.values.get('distill-io'), flash=flash, datastore=datastore)
|
d_importer.run(data=request.values.get('distill-io'), flash=flash, datastore=datastore)
|
||||||
for uuid in d_importer.new_uuids:
|
for uuid in d_importer.new_uuids:
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': True}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
|
|
||||||
# XLSX importer
|
# XLSX importer
|
||||||
if request.files and request.files.get('xlsx_file'):
|
if request.files and request.files.get('xlsx_file'):
|
||||||
@@ -1017,7 +1014,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
w_importer.run(data=file, flash=flash, datastore=datastore)
|
w_importer.run(data=file, flash=flash, datastore=datastore)
|
||||||
|
|
||||||
for uuid in w_importer.new_uuids:
|
for uuid in w_importer.new_uuids:
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': True}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
|
|
||||||
# Could be some remaining, or we could be on GET
|
# Could be some remaining, or we could be on GET
|
||||||
form = forms.importForm(formdata=request.form if request.method == 'POST' else None)
|
form = forms.importForm(formdata=request.form if request.method == 'POST' else None)
|
||||||
@@ -1158,8 +1155,6 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
@login_optionally_required
|
@login_optionally_required
|
||||||
def preview_page(uuid):
|
def preview_page(uuid):
|
||||||
content = []
|
content = []
|
||||||
ignored_line_numbers = []
|
|
||||||
trigger_line_numbers = []
|
|
||||||
versions = []
|
versions = []
|
||||||
timestamp = None
|
timestamp = None
|
||||||
|
|
||||||
@@ -1176,11 +1171,10 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
system_uses_webdriver = datastore.data['settings']['application']['fetch_backend'] == 'html_webdriver'
|
system_uses_webdriver = datastore.data['settings']['application']['fetch_backend'] == 'html_webdriver'
|
||||||
extra_stylesheets = [url_for('static_content', group='styles', filename='diff.css')]
|
extra_stylesheets = [url_for('static_content', group='styles', filename='diff.css')]
|
||||||
|
|
||||||
|
|
||||||
is_html_webdriver = False
|
is_html_webdriver = False
|
||||||
if (watch.get('fetch_backend') == 'system' and system_uses_webdriver) or watch.get('fetch_backend') == 'html_webdriver' or watch.get('fetch_backend', '').startswith('extra_browser_'):
|
if (watch.get('fetch_backend') == 'system' and system_uses_webdriver) or watch.get('fetch_backend') == 'html_webdriver' or watch.get('fetch_backend', '').startswith('extra_browser_'):
|
||||||
is_html_webdriver = True
|
is_html_webdriver = True
|
||||||
|
triggered_line_numbers = []
|
||||||
if datastore.data['watching'][uuid].history_n == 0 and (watch.get_error_text() or watch.get_error_snapshot()):
|
if datastore.data['watching'][uuid].history_n == 0 and (watch.get_error_text() or watch.get_error_snapshot()):
|
||||||
flash("Preview unavailable - No fetch/check completed or triggers not reached", "error")
|
flash("Preview unavailable - No fetch/check completed or triggers not reached", "error")
|
||||||
else:
|
else:
|
||||||
@@ -1193,31 +1187,12 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
versions = list(watch.history.keys())
|
versions = list(watch.history.keys())
|
||||||
tmp = watch.get_history_snapshot(timestamp).splitlines()
|
content = watch.get_history_snapshot(timestamp)
|
||||||
|
|
||||||
# Get what needs to be highlighted
|
triggered_line_numbers = html_tools.strip_ignore_text(content=content,
|
||||||
ignore_rules = watch.get('ignore_text', []) + datastore.data['settings']['application']['global_ignore_text']
|
wordlist=watch['trigger_text'],
|
||||||
|
mode='line numbers'
|
||||||
# .readlines will keep the \n, but we will parse it here again, in the future tidy this up
|
)
|
||||||
ignored_line_numbers = html_tools.strip_ignore_text(content="\n".join(tmp),
|
|
||||||
wordlist=ignore_rules,
|
|
||||||
mode='line numbers'
|
|
||||||
)
|
|
||||||
|
|
||||||
trigger_line_numbers = html_tools.strip_ignore_text(content="\n".join(tmp),
|
|
||||||
wordlist=watch['trigger_text'],
|
|
||||||
mode='line numbers'
|
|
||||||
)
|
|
||||||
# Prepare the classes and lines used in the template
|
|
||||||
i=0
|
|
||||||
for l in tmp:
|
|
||||||
classes=[]
|
|
||||||
i+=1
|
|
||||||
if i in ignored_line_numbers:
|
|
||||||
classes.append('ignored')
|
|
||||||
if i in trigger_line_numbers:
|
|
||||||
classes.append('triggered')
|
|
||||||
content.append({'line': l, 'classes': ' '.join(classes)})
|
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
content.append({'line': f"File doesnt exist or unable to read timestamp {timestamp}", 'classes': ''})
|
content.append({'line': f"File doesnt exist or unable to read timestamp {timestamp}", 'classes': ''})
|
||||||
@@ -1228,8 +1203,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
history_n=watch.history_n,
|
history_n=watch.history_n,
|
||||||
extra_stylesheets=extra_stylesheets,
|
extra_stylesheets=extra_stylesheets,
|
||||||
extra_title=f" - Diff - {watch.label} @ {timestamp}",
|
extra_title=f" - Diff - {watch.label} @ {timestamp}",
|
||||||
ignored_line_numbers=ignored_line_numbers,
|
triggered_line_numbers=triggered_line_numbers,
|
||||||
triggered_line_numbers=trigger_line_numbers,
|
|
||||||
current_diff_url=watch['url'],
|
current_diff_url=watch['url'],
|
||||||
screenshot=watch.get_screenshot(),
|
screenshot=watch.get_screenshot(),
|
||||||
watch=watch,
|
watch=watch,
|
||||||
@@ -1400,55 +1374,13 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
# Return a 500 error
|
# Return a 500 error
|
||||||
abort(500)
|
abort(500)
|
||||||
|
|
||||||
|
# Ajax callback
|
||||||
@app.route("/edit/<string:uuid>/preview-rendered", methods=['POST'])
|
@app.route("/edit/<string:uuid>/preview-rendered", methods=['POST'])
|
||||||
@login_optionally_required
|
@login_optionally_required
|
||||||
def watch_get_preview_rendered(uuid):
|
def watch_get_preview_rendered(uuid):
|
||||||
'''For when viewing the "preview" of the rendered text from inside of Edit'''
|
'''For when viewing the "preview" of the rendered text from inside of Edit'''
|
||||||
now = time.time()
|
from .processors.text_json_diff import prepare_filter_prevew
|
||||||
import brotli
|
return prepare_filter_prevew(watch_uuid=uuid, datastore=datastore)
|
||||||
from . import forms
|
|
||||||
|
|
||||||
text_after_filter = ''
|
|
||||||
tmp_watch = deepcopy(datastore.data['watching'].get(uuid))
|
|
||||||
|
|
||||||
if tmp_watch and tmp_watch.history and os.path.isdir(tmp_watch.watch_data_dir):
|
|
||||||
# Splice in the temporary stuff from the form
|
|
||||||
form = forms.processor_text_json_diff_form(formdata=request.form if request.method == 'POST' else None,
|
|
||||||
data=request.form
|
|
||||||
)
|
|
||||||
# Only update vars that came in via the AJAX post
|
|
||||||
p = {k: v for k, v in form.data.items() if k in request.form.keys()}
|
|
||||||
tmp_watch.update(p)
|
|
||||||
|
|
||||||
latest_filename = next(reversed(tmp_watch.history))
|
|
||||||
html_fname = os.path.join(tmp_watch.watch_data_dir, f"{latest_filename}.html.br")
|
|
||||||
with open(html_fname, 'rb') as f:
|
|
||||||
decompressed_data = brotli.decompress(f.read()).decode('utf-8') if html_fname.endswith('.br') else f.read().decode('utf-8')
|
|
||||||
|
|
||||||
# Just like a normal change detection except provide a fake "watch" object and dont call .call_browser()
|
|
||||||
processor_module = importlib.import_module("changedetectionio.processors.text_json_diff.processor")
|
|
||||||
update_handler = processor_module.perform_site_check(datastore=datastore,
|
|
||||||
watch_uuid=uuid # probably not needed anymore anyway?
|
|
||||||
)
|
|
||||||
# Use the last loaded HTML as the input
|
|
||||||
update_handler.fetcher.content = decompressed_data
|
|
||||||
try:
|
|
||||||
changed_detected, update_obj, contents, text_after_filter = update_handler.run_changedetection(
|
|
||||||
watch=tmp_watch,
|
|
||||||
skip_when_checksum_same=False,
|
|
||||||
)
|
|
||||||
except FilterNotFoundInResponse as e:
|
|
||||||
text_after_filter = f"Filter not found in HTML: {str(e)}"
|
|
||||||
except ReplyWithContentButNoText as e:
|
|
||||||
text_after_filter = f"Filter found but no text (empty result)"
|
|
||||||
except Exception as e:
|
|
||||||
text_after_filter = f"Error: {str(e)}"
|
|
||||||
|
|
||||||
if not text_after_filter.strip():
|
|
||||||
text_after_filter = 'Empty content'
|
|
||||||
|
|
||||||
logger.trace(f"Parsed in {time.time()-now:.3f}s")
|
|
||||||
return text_after_filter.strip()
|
|
||||||
|
|
||||||
|
|
||||||
@app.route("/form/add/quickwatch", methods=['POST'])
|
@app.route("/form/add/quickwatch", methods=['POST'])
|
||||||
@@ -1465,7 +1397,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
url = request.form.get('url').strip()
|
url = request.form.get('url').strip()
|
||||||
if datastore.url_exists(url):
|
if datastore.url_exists(url):
|
||||||
flash(f'Warning, URL {url} already exists', "notice")
|
flash(f'Warning, URL {url} already exists', "notice")
|
||||||
|
|
||||||
add_paused = request.form.get('edit_and_watch_submit_button') != None
|
add_paused = request.form.get('edit_and_watch_submit_button') != None
|
||||||
processor = request.form.get('processor', 'text_json_diff')
|
processor = request.form.get('processor', 'text_json_diff')
|
||||||
new_uuid = datastore.add_watch(url=url, tag=request.form.get('tags').strip(), extras={'paused': add_paused, 'processor': processor})
|
new_uuid = datastore.add_watch(url=url, tag=request.form.get('tags').strip(), extras={'paused': add_paused, 'processor': processor})
|
||||||
@@ -1511,7 +1443,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
new_uuid = datastore.clone(uuid)
|
new_uuid = datastore.clone(uuid)
|
||||||
if new_uuid:
|
if new_uuid:
|
||||||
if not datastore.data['watching'].get(uuid).get('paused'):
|
if not datastore.data['watching'].get(uuid).get('paused'):
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=5, item={'uuid': new_uuid, 'skip_when_checksum_same': True}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=5, item={'uuid': new_uuid}))
|
||||||
flash('Cloned.')
|
flash('Cloned.')
|
||||||
|
|
||||||
return redirect(url_for('index'))
|
return redirect(url_for('index'))
|
||||||
@@ -1532,7 +1464,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
|
|
||||||
if uuid:
|
if uuid:
|
||||||
if uuid not in running_uuids:
|
if uuid not in running_uuids:
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': False}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
i = 1
|
i = 1
|
||||||
|
|
||||||
elif tag:
|
elif tag:
|
||||||
@@ -1543,7 +1475,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
continue
|
continue
|
||||||
if watch_uuid not in running_uuids and not datastore.data['watching'][watch_uuid]['paused']:
|
if watch_uuid not in running_uuids and not datastore.data['watching'][watch_uuid]['paused']:
|
||||||
update_q.put(
|
update_q.put(
|
||||||
queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': watch_uuid, 'skip_when_checksum_same': False})
|
queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': watch_uuid})
|
||||||
)
|
)
|
||||||
i += 1
|
i += 1
|
||||||
|
|
||||||
@@ -1553,9 +1485,8 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
if watch_uuid not in running_uuids and not datastore.data['watching'][watch_uuid]['paused']:
|
if watch_uuid not in running_uuids and not datastore.data['watching'][watch_uuid]['paused']:
|
||||||
if with_errors and not watch.get('last_error'):
|
if with_errors and not watch.get('last_error'):
|
||||||
continue
|
continue
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': watch_uuid, 'skip_when_checksum_same': False}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': watch_uuid}))
|
||||||
i += 1
|
i += 1
|
||||||
|
|
||||||
flash(f"{i} watches queued for rechecking.")
|
flash(f"{i} watches queued for rechecking.")
|
||||||
return redirect(url_for('index', tag=tag))
|
return redirect(url_for('index', tag=tag))
|
||||||
|
|
||||||
@@ -1612,7 +1543,7 @@ def changedetection_app(config=None, datastore_o=None):
|
|||||||
uuid = uuid.strip()
|
uuid = uuid.strip()
|
||||||
if datastore.data['watching'].get(uuid):
|
if datastore.data['watching'].get(uuid):
|
||||||
# Recheck and require a full reprocessing
|
# Recheck and require a full reprocessing
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid, 'skip_when_checksum_same': False}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=1, item={'uuid': uuid}))
|
||||||
flash("{} watches queued for rechecking".format(len(uuids)))
|
flash("{} watches queued for rechecking".format(len(uuids)))
|
||||||
|
|
||||||
elif (op == 'clear-errors'):
|
elif (op == 'clear-errors'):
|
||||||
@@ -1936,7 +1867,7 @@ def ticker_thread_check_time_launch_checks():
|
|||||||
f"{now - watch['last_checked']:0.2f}s since last checked")
|
f"{now - watch['last_checked']:0.2f}s since last checked")
|
||||||
|
|
||||||
# Into the queue with you
|
# Into the queue with you
|
||||||
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=priority, item={'uuid': uuid, 'skip_when_checksum_same': True}))
|
update_q.put(queuedWatchMetaData.PrioritizedItem(priority=priority, item={'uuid': uuid}))
|
||||||
|
|
||||||
# Reset for next time
|
# Reset for next time
|
||||||
watch.jitter_seconds = 0
|
watch.jitter_seconds = 0
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
from loguru import logger
|
||||||
|
|
||||||
from changedetectionio.strtobool import strtobool
|
from changedetectionio.strtobool import strtobool
|
||||||
|
|
||||||
@@ -475,7 +476,7 @@ class processor_text_json_diff_form(commonSettingsForm):
|
|||||||
|
|
||||||
title = StringField('Title', default='')
|
title = StringField('Title', default='')
|
||||||
|
|
||||||
ignore_text = StringListField('Ignore text', [ValidateListRegex()])
|
ignore_text = StringListField('Ignore lines containing', [ValidateListRegex()])
|
||||||
headers = StringDictKeyValue('Request headers')
|
headers = StringDictKeyValue('Request headers')
|
||||||
body = TextAreaField('Request body', [validators.Optional()])
|
body = TextAreaField('Request body', [validators.Optional()])
|
||||||
method = SelectField('Request method', choices=valid_method, default=default_method)
|
method = SelectField('Request method', choices=valid_method, default=default_method)
|
||||||
@@ -525,9 +526,16 @@ class processor_text_json_diff_form(commonSettingsForm):
|
|||||||
try:
|
try:
|
||||||
from changedetectionio.safe_jinja import render as jinja_render
|
from changedetectionio.safe_jinja import render as jinja_render
|
||||||
jinja_render(template_str=self.url.data)
|
jinja_render(template_str=self.url.data)
|
||||||
|
except ModuleNotFoundError as e:
|
||||||
|
# incase jinja2_time or others is missing
|
||||||
|
logger.error(e)
|
||||||
|
self.url.errors.append(e)
|
||||||
|
result = False
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
logger.error(e)
|
||||||
self.url.errors.append('Invalid template syntax')
|
self.url.errors.append('Invalid template syntax')
|
||||||
result = False
|
result = False
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
class SingleExtraProxy(Form):
|
class SingleExtraProxy(Form):
|
||||||
@@ -580,6 +588,8 @@ class globalSettingsApplicationForm(commonSettingsForm):
|
|||||||
global_ignore_text = StringListField('Ignore Text', [ValidateListRegex()])
|
global_ignore_text = StringListField('Ignore Text', [ValidateListRegex()])
|
||||||
global_subtractive_selectors = StringListField('Remove elements', [ValidateCSSJSONXPATHInput(allow_json=False)])
|
global_subtractive_selectors = StringListField('Remove elements', [ValidateCSSJSONXPATHInput(allow_json=False)])
|
||||||
ignore_whitespace = BooleanField('Ignore whitespace')
|
ignore_whitespace = BooleanField('Ignore whitespace')
|
||||||
|
keep_history_n = IntegerField('Number of snapshots to keep in history for each watch')
|
||||||
|
keep_history_seconds = IntegerField('Number of snapshots to keep - maximum age (todo/seconds)')
|
||||||
password = SaltyPasswordField()
|
password = SaltyPasswordField()
|
||||||
pager_size = IntegerField('Pager size',
|
pager_size = IntegerField('Pager size',
|
||||||
render_kw={"style": "width: 5em;"},
|
render_kw={"style": "width: 5em;"},
|
||||||
|
|||||||
@@ -3,11 +3,11 @@ from lxml import etree
|
|||||||
import json
|
import json
|
||||||
import re
|
import re
|
||||||
|
|
||||||
|
|
||||||
# HTML added to be sure each result matching a filter (.example) gets converted to a new line by Inscriptis
|
# HTML added to be sure each result matching a filter (.example) gets converted to a new line by Inscriptis
|
||||||
TEXT_FILTER_LIST_LINE_SUFFIX = "<br>"
|
TEXT_FILTER_LIST_LINE_SUFFIX = "<br>"
|
||||||
|
TRANSLATE_WHITESPACE_TABLE = str.maketrans('', '', '\r\n\t ')
|
||||||
PERL_STYLE_REGEX = r'^/(.*?)/([a-z]*)?$'
|
PERL_STYLE_REGEX = r'^/(.*?)/([a-z]*)?$'
|
||||||
|
|
||||||
# 'price' , 'lowPrice', 'highPrice' are usually under here
|
# 'price' , 'lowPrice', 'highPrice' are usually under here
|
||||||
# All of those may or may not appear on different websites - I didnt find a way todo case-insensitive searching here
|
# All of those may or may not appear on different websites - I didnt find a way todo case-insensitive searching here
|
||||||
LD_JSON_PRODUCT_OFFER_SELECTORS = ["json:$..offers", "json:$..Offers"]
|
LD_JSON_PRODUCT_OFFER_SELECTORS = ["json:$..offers", "json:$..Offers"]
|
||||||
@@ -326,6 +326,7 @@ def extract_json_as_string(content, json_filter, ensure_is_ldjson_info_type=None
|
|||||||
# - "line numbers" return a list of line numbers that match (int list)
|
# - "line numbers" return a list of line numbers that match (int list)
|
||||||
#
|
#
|
||||||
# wordlist - list of regex's (str) or words (str)
|
# wordlist - list of regex's (str) or words (str)
|
||||||
|
# Preserves all linefeeds and other whitespacing, its not the job of this to remove that
|
||||||
def strip_ignore_text(content, wordlist, mode="content"):
|
def strip_ignore_text(content, wordlist, mode="content"):
|
||||||
i = 0
|
i = 0
|
||||||
output = []
|
output = []
|
||||||
@@ -341,32 +342,30 @@ def strip_ignore_text(content, wordlist, mode="content"):
|
|||||||
else:
|
else:
|
||||||
ignore_text.append(k.strip())
|
ignore_text.append(k.strip())
|
||||||
|
|
||||||
for line in content.splitlines():
|
for line in content.splitlines(keepends=True):
|
||||||
i += 1
|
i += 1
|
||||||
# Always ignore blank lines in this mode. (when this function gets called)
|
# Always ignore blank lines in this mode. (when this function gets called)
|
||||||
got_match = False
|
got_match = False
|
||||||
if len(line.strip()):
|
for l in ignore_text:
|
||||||
for l in ignore_text:
|
if l.lower() in line.lower():
|
||||||
if l.lower() in line.lower():
|
got_match = True
|
||||||
|
|
||||||
|
if not got_match:
|
||||||
|
for r in ignore_regex:
|
||||||
|
if r.search(line):
|
||||||
got_match = True
|
got_match = True
|
||||||
|
|
||||||
if not got_match:
|
if not got_match:
|
||||||
for r in ignore_regex:
|
# Not ignored, and should preserve "keepends"
|
||||||
if r.search(line):
|
output.append(line)
|
||||||
got_match = True
|
else:
|
||||||
|
ignored_line_numbers.append(i)
|
||||||
if not got_match:
|
|
||||||
# Not ignored
|
|
||||||
output.append(line.encode('utf8'))
|
|
||||||
else:
|
|
||||||
ignored_line_numbers.append(i)
|
|
||||||
|
|
||||||
|
|
||||||
# Used for finding out what to highlight
|
# Used for finding out what to highlight
|
||||||
if mode == "line numbers":
|
if mode == "line numbers":
|
||||||
return ignored_line_numbers
|
return ignored_line_numbers
|
||||||
|
|
||||||
return "\n".encode('utf8').join(output)
|
return ''.join(output)
|
||||||
|
|
||||||
def cdata_in_document_to_text(html_content: str, render_anchor_tag_content=False) -> str:
|
def cdata_in_document_to_text(html_content: str, render_anchor_tag_content=False) -> str:
|
||||||
from xml.sax.saxutils import escape as xml_escape
|
from xml.sax.saxutils import escape as xml_escape
|
||||||
|
|||||||
@@ -40,6 +40,8 @@ class model(dict):
|
|||||||
'global_ignore_text': [], # List of text to ignore when calculating the comparison checksum
|
'global_ignore_text': [], # List of text to ignore when calculating the comparison checksum
|
||||||
'global_subtractive_selectors': [],
|
'global_subtractive_selectors': [],
|
||||||
'ignore_whitespace': True,
|
'ignore_whitespace': True,
|
||||||
|
'keep_history_n': None, # Number of snapshots to keep
|
||||||
|
'keep_history_seconds': None, # Or time ago back to keep
|
||||||
'notification_body': default_notification_body,
|
'notification_body': default_notification_body,
|
||||||
'notification_format': default_notification_format,
|
'notification_format': default_notification_format,
|
||||||
'notification_title': default_notification_title,
|
'notification_title': default_notification_title,
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ import re
|
|||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
|
|
||||||
|
from ..html_tools import TRANSLATE_WHITESPACE_TABLE
|
||||||
|
|
||||||
# Allowable protocols, protects against javascript: etc
|
# Allowable protocols, protects against javascript: etc
|
||||||
# file:// is further checked by ALLOW_FILE_URI
|
# file:// is further checked by ALLOW_FILE_URI
|
||||||
SAFE_PROTOCOL_REGEX='^(http|https|ftp|file):'
|
SAFE_PROTOCOL_REGEX='^(http|https|ftp|file):'
|
||||||
@@ -36,8 +38,9 @@ class model(watch_base):
|
|||||||
jitter_seconds = 0
|
jitter_seconds = 0
|
||||||
|
|
||||||
def __init__(self, *arg, **kw):
|
def __init__(self, *arg, **kw):
|
||||||
self.__datastore_path = kw['datastore_path']
|
self.__datastore_path = kw.get('datastore_path')
|
||||||
del kw['datastore_path']
|
if kw.get('datastore_path'):
|
||||||
|
del kw['datastore_path']
|
||||||
super(model, self).__init__(*arg, **kw)
|
super(model, self).__init__(*arg, **kw)
|
||||||
if kw.get('default'):
|
if kw.get('default'):
|
||||||
self.update(kw['default'])
|
self.update(kw['default'])
|
||||||
@@ -171,6 +174,10 @@ class model(watch_base):
|
|||||||
"""
|
"""
|
||||||
tmp_history = {}
|
tmp_history = {}
|
||||||
|
|
||||||
|
# In the case we are only using the watch for processing without history
|
||||||
|
if not self.watch_data_dir:
|
||||||
|
return []
|
||||||
|
|
||||||
# Read the history file as a dict
|
# Read the history file as a dict
|
||||||
fname = os.path.join(self.watch_data_dir, "history.txt")
|
fname = os.path.join(self.watch_data_dir, "history.txt")
|
||||||
if os.path.isfile(fname):
|
if os.path.isfile(fname):
|
||||||
@@ -307,13 +314,13 @@ class model(watch_base):
|
|||||||
dest = os.path.join(self.watch_data_dir, snapshot_fname)
|
dest = os.path.join(self.watch_data_dir, snapshot_fname)
|
||||||
if not os.path.exists(dest):
|
if not os.path.exists(dest):
|
||||||
with open(dest, 'wb') as f:
|
with open(dest, 'wb') as f:
|
||||||
f.write(brotli.compress(contents, mode=brotli.MODE_TEXT))
|
f.write(brotli.compress(contents.encode('utf-8'), mode=brotli.MODE_TEXT))
|
||||||
else:
|
else:
|
||||||
snapshot_fname = f"{snapshot_id}.txt"
|
snapshot_fname = f"{snapshot_id}.txt"
|
||||||
dest = os.path.join(self.watch_data_dir, snapshot_fname)
|
dest = os.path.join(self.watch_data_dir, snapshot_fname)
|
||||||
if not os.path.exists(dest):
|
if not os.path.exists(dest):
|
||||||
with open(dest, 'wb') as f:
|
with open(dest, 'wb') as f:
|
||||||
f.write(contents)
|
f.write(contents.encode('utf-8'))
|
||||||
|
|
||||||
# Append to index
|
# Append to index
|
||||||
# @todo check last char was \n
|
# @todo check last char was \n
|
||||||
@@ -345,14 +352,32 @@ class model(watch_base):
|
|||||||
return seconds
|
return seconds
|
||||||
|
|
||||||
# Iterate over all history texts and see if something new exists
|
# Iterate over all history texts and see if something new exists
|
||||||
def lines_contain_something_unique_compared_to_history(self, lines: list):
|
# Always applying .strip() to start/end but optionally replace any other whitespace
|
||||||
local_lines = set([l.decode('utf-8').strip().lower() for l in lines])
|
def lines_contain_something_unique_compared_to_history(self, lines: list, ignore_whitespace=False):
|
||||||
|
local_lines = []
|
||||||
|
if lines:
|
||||||
|
if ignore_whitespace:
|
||||||
|
if isinstance(lines[0], str): # Can be either str or bytes depending on what was on the disk
|
||||||
|
local_lines = set([l.translate(TRANSLATE_WHITESPACE_TABLE).lower() for l in lines])
|
||||||
|
else:
|
||||||
|
local_lines = set([l.decode('utf-8').translate(TRANSLATE_WHITESPACE_TABLE).lower() for l in lines])
|
||||||
|
else:
|
||||||
|
if isinstance(lines[0], str): # Can be either str or bytes depending on what was on the disk
|
||||||
|
local_lines = set([l.strip().lower() for l in lines])
|
||||||
|
else:
|
||||||
|
local_lines = set([l.decode('utf-8').strip().lower() for l in lines])
|
||||||
|
|
||||||
|
|
||||||
# Compare each lines (set) against each history text file (set) looking for something new..
|
# Compare each lines (set) against each history text file (set) looking for something new..
|
||||||
existing_history = set({})
|
existing_history = set({})
|
||||||
for k, v in self.history.items():
|
for k, v in self.history.items():
|
||||||
content = self.get_history_snapshot(k)
|
content = self.get_history_snapshot(k)
|
||||||
alist = set([line.strip().lower() for line in content.splitlines()])
|
|
||||||
|
if ignore_whitespace:
|
||||||
|
alist = set([line.translate(TRANSLATE_WHITESPACE_TABLE).lower() for line in content.splitlines()])
|
||||||
|
else:
|
||||||
|
alist = set([line.strip().lower() for line in content.splitlines()])
|
||||||
|
|
||||||
existing_history = existing_history.union(alist)
|
existing_history = existing_history.union(alist)
|
||||||
|
|
||||||
# Check that everything in local_lines(new stuff) already exists in existing_history - it should
|
# Check that everything in local_lines(new stuff) already exists in existing_history - it should
|
||||||
@@ -396,8 +421,8 @@ class model(watch_base):
|
|||||||
@property
|
@property
|
||||||
def watch_data_dir(self):
|
def watch_data_dir(self):
|
||||||
# The base dir of the watch data
|
# The base dir of the watch data
|
||||||
return os.path.join(self.__datastore_path, self['uuid'])
|
return os.path.join(self.__datastore_path, self['uuid']) if self.__datastore_path else None
|
||||||
|
|
||||||
def get_error_text(self):
|
def get_error_text(self):
|
||||||
"""Return the text saved from a previous request that resulted in a non-200 error"""
|
"""Return the text saved from a previous request that resulted in a non-200 error"""
|
||||||
fname = os.path.join(self.watch_data_dir, "last-error.txt")
|
fname = os.path.join(self.watch_data_dir, "last-error.txt")
|
||||||
@@ -600,6 +625,9 @@ class model(watch_base):
|
|||||||
if index > 1 and os.path.isfile(filepath):
|
if index > 1 and os.path.isfile(filepath):
|
||||||
os.remove(filepath)
|
os.remove(filepath)
|
||||||
|
|
||||||
|
def post_process(self):
|
||||||
|
|
||||||
|
x=1
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def get_browsersteps_available_screenshots(self):
|
def get_browsersteps_available_screenshots(self):
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ class watch_base(dict):
|
|||||||
'check_count': 0,
|
'check_count': 0,
|
||||||
'check_unique_lines': False, # On change-detected, compare against all history if its something new
|
'check_unique_lines': False, # On change-detected, compare against all history if its something new
|
||||||
'consecutive_filter_failures': 0, # Every time the CSS/xPath filter cannot be located, reset when all is fine.
|
'consecutive_filter_failures': 0, # Every time the CSS/xPath filter cannot be located, reset when all is fine.
|
||||||
|
'content-type': None,
|
||||||
'date_created': None,
|
'date_created': None,
|
||||||
'extract_text': [], # Extract text by regex after filters
|
'extract_text': [], # Extract text by regex after filters
|
||||||
'extract_title_as_title': False,
|
'extract_title_as_title': False,
|
||||||
@@ -32,6 +33,8 @@ class watch_base(dict):
|
|||||||
'headers': {}, # Extra headers to send
|
'headers': {}, # Extra headers to send
|
||||||
'ignore_text': [], # List of text to ignore when calculating the comparison checksum
|
'ignore_text': [], # List of text to ignore when calculating the comparison checksum
|
||||||
'in_stock_only': True, # Only trigger change on going to instock from out-of-stock
|
'in_stock_only': True, # Only trigger change on going to instock from out-of-stock
|
||||||
|
'keep_history_n': None, # Number of snapshots to keep
|
||||||
|
'keep_history_seconds': None, # Or time ago back to keep
|
||||||
'include_filters': [],
|
'include_filters': [],
|
||||||
'last_checked': 0,
|
'last_checked': 0,
|
||||||
'last_error': False,
|
'last_error': False,
|
||||||
|
|||||||
@@ -1,16 +1,14 @@
|
|||||||
from abc import abstractmethod
|
from abc import abstractmethod
|
||||||
|
|
||||||
from changedetectionio.content_fetchers.base import Fetcher
|
from changedetectionio.content_fetchers.base import Fetcher
|
||||||
from changedetectionio.strtobool import strtobool
|
from changedetectionio.strtobool import strtobool
|
||||||
|
|
||||||
from copy import deepcopy
|
from copy import deepcopy
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
import hashlib
|
import hashlib
|
||||||
import os
|
|
||||||
import re
|
|
||||||
import importlib
|
import importlib
|
||||||
import pkgutil
|
|
||||||
import inspect
|
import inspect
|
||||||
|
import os
|
||||||
|
import pkgutil
|
||||||
|
import re
|
||||||
|
|
||||||
class difference_detection_processor():
|
class difference_detection_processor():
|
||||||
|
|
||||||
@@ -20,6 +18,7 @@ class difference_detection_processor():
|
|||||||
screenshot = None
|
screenshot = None
|
||||||
watch = None
|
watch = None
|
||||||
xpath_data = None
|
xpath_data = None
|
||||||
|
preferred_proxy = None
|
||||||
|
|
||||||
def __init__(self, *args, datastore, watch_uuid, **kwargs):
|
def __init__(self, *args, datastore, watch_uuid, **kwargs):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
@@ -28,7 +27,8 @@ class difference_detection_processor():
|
|||||||
# Generic fetcher that should be extended (requests, playwright etc)
|
# Generic fetcher that should be extended (requests, playwright etc)
|
||||||
self.fetcher = Fetcher()
|
self.fetcher = Fetcher()
|
||||||
|
|
||||||
def call_browser(self):
|
def call_browser(self, preferred_proxy_id=None):
|
||||||
|
|
||||||
from requests.structures import CaseInsensitiveDict
|
from requests.structures import CaseInsensitiveDict
|
||||||
|
|
||||||
# Protect against file:// access
|
# Protect against file:// access
|
||||||
@@ -44,7 +44,7 @@ class difference_detection_processor():
|
|||||||
prefer_fetch_backend = self.watch.get('fetch_backend', 'system')
|
prefer_fetch_backend = self.watch.get('fetch_backend', 'system')
|
||||||
|
|
||||||
# Proxy ID "key"
|
# Proxy ID "key"
|
||||||
preferred_proxy_id = self.datastore.get_preferred_proxy_for_watch(uuid=self.watch.get('uuid'))
|
preferred_proxy_id = preferred_proxy_id if preferred_proxy_id else self.datastore.get_preferred_proxy_for_watch(uuid=self.watch.get('uuid'))
|
||||||
|
|
||||||
# Pluggable content self.fetcher
|
# Pluggable content self.fetcher
|
||||||
if not prefer_fetch_backend or prefer_fetch_backend == 'system':
|
if not prefer_fetch_backend or prefer_fetch_backend == 'system':
|
||||||
@@ -157,12 +157,12 @@ class difference_detection_processor():
|
|||||||
# After init, call run_changedetection() which will do the actual change-detection
|
# After init, call run_changedetection() which will do the actual change-detection
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def run_changedetection(self, watch, skip_when_checksum_same=True):
|
def run_changedetection(self, watch):
|
||||||
update_obj = {'last_notification_error': False, 'last_error': False}
|
update_obj = {'last_notification_error': False, 'last_error': False}
|
||||||
some_data = 'xxxxx'
|
some_data = 'xxxxx'
|
||||||
update_obj["previous_md5"] = hashlib.md5(some_data.encode('utf-8')).hexdigest()
|
update_obj["previous_md5"] = hashlib.md5(some_data.encode('utf-8')).hexdigest()
|
||||||
changed_detected = False
|
changed_detected = False
|
||||||
return changed_detected, update_obj, ''.encode('utf-8'), b''
|
return changed_detected, update_obj, ''.encode('utf-8')
|
||||||
|
|
||||||
|
|
||||||
def find_sub_packages(package_name):
|
def find_sub_packages(package_name):
|
||||||
|
|||||||
@@ -27,22 +27,27 @@ def _search_prop_by_value(matches, value):
|
|||||||
return prop[1] # Yield the desired value and exit the function
|
return prop[1] # Yield the desired value and exit the function
|
||||||
|
|
||||||
def _deduplicate_prices(data):
|
def _deduplicate_prices(data):
|
||||||
seen = set()
|
import re
|
||||||
unique_data = []
|
|
||||||
|
|
||||||
|
'''
|
||||||
|
Some price data has multiple entries, OR it has a single entry with ['$159', '159', 159, "$ 159"] or just "159"
|
||||||
|
Get all the values, clean it and add it to a set then return the unique values
|
||||||
|
'''
|
||||||
|
unique_data = set()
|
||||||
|
|
||||||
|
# Return the complete 'datum' where its price was not seen before
|
||||||
for datum in data:
|
for datum in data:
|
||||||
# Convert 'value' to float if it can be a numeric string, otherwise leave it as is
|
|
||||||
try:
|
|
||||||
normalized_value = float(datum.value) if isinstance(datum.value, str) and datum.value.replace('.', '', 1).isdigit() else datum.value
|
|
||||||
except ValueError:
|
|
||||||
normalized_value = datum.value
|
|
||||||
|
|
||||||
# If the normalized value hasn't been seen yet, add it to unique data
|
if isinstance(datum.value, list):
|
||||||
if normalized_value not in seen:
|
# Process each item in the list
|
||||||
unique_data.append(datum)
|
normalized_value = set([float(re.sub(r'[^\d.]', '', str(item))) for item in datum.value])
|
||||||
seen.add(normalized_value)
|
unique_data.update(normalized_value)
|
||||||
|
else:
|
||||||
return unique_data
|
# Process single value
|
||||||
|
v = float(re.sub(r'[^\d.]', '', str(datum.value)))
|
||||||
|
unique_data.add(v)
|
||||||
|
|
||||||
|
return list(unique_data)
|
||||||
|
|
||||||
|
|
||||||
# should return Restock()
|
# should return Restock()
|
||||||
@@ -83,14 +88,13 @@ def get_itemprop_availability(html_content) -> Restock:
|
|||||||
if price_result:
|
if price_result:
|
||||||
# Right now, we just support single product items, maybe we will store the whole actual metadata seperately in teh future and
|
# Right now, we just support single product items, maybe we will store the whole actual metadata seperately in teh future and
|
||||||
# parse that for the UI?
|
# parse that for the UI?
|
||||||
prices_found = set(str(item.value).replace('$', '') for item in price_result)
|
if len(price_result) > 1 and len(price_result) > 1:
|
||||||
if len(price_result) > 1 and len(prices_found) > 1:
|
|
||||||
# See of all prices are different, in the case that one product has many embedded data types with the same price
|
# See of all prices are different, in the case that one product has many embedded data types with the same price
|
||||||
# One might have $121.95 and another 121.95 etc
|
# One might have $121.95 and another 121.95 etc
|
||||||
logger.warning(f"More than one price found {prices_found}, throwing exception, cant use this plugin.")
|
logger.warning(f"More than one price found {price_result}, throwing exception, cant use this plugin.")
|
||||||
raise MoreThanOnePriceFound()
|
raise MoreThanOnePriceFound()
|
||||||
|
|
||||||
value['price'] = price_result[0].value
|
value['price'] = price_result[0]
|
||||||
|
|
||||||
pricecurrency_result = pricecurrency_parse.find(data)
|
pricecurrency_result = pricecurrency_parse.find(data)
|
||||||
if pricecurrency_result:
|
if pricecurrency_result:
|
||||||
@@ -140,7 +144,7 @@ class perform_site_check(difference_detection_processor):
|
|||||||
screenshot = None
|
screenshot = None
|
||||||
xpath_data = None
|
xpath_data = None
|
||||||
|
|
||||||
def run_changedetection(self, watch, skip_when_checksum_same=True):
|
def run_changedetection(self, watch):
|
||||||
import hashlib
|
import hashlib
|
||||||
|
|
||||||
if not watch:
|
if not watch:
|
||||||
@@ -220,7 +224,7 @@ class perform_site_check(difference_detection_processor):
|
|||||||
itemprop_availability['original_price'] = itemprop_availability.get('price')
|
itemprop_availability['original_price'] = itemprop_availability.get('price')
|
||||||
update_obj['restock']["original_price"] = itemprop_availability.get('price')
|
update_obj['restock']["original_price"] = itemprop_availability.get('price')
|
||||||
|
|
||||||
if not self.fetcher.instock_data and not itemprop_availability.get('availability'):
|
if not self.fetcher.instock_data and not itemprop_availability.get('availability') and not itemprop_availability.get('price'):
|
||||||
raise ProcessorException(
|
raise ProcessorException(
|
||||||
message=f"Unable to extract restock data for this page unfortunately. (Got code {self.fetcher.get_last_status_code()} from server), no embedded stock information was found and nothing interesting in the text, try using this watch with Chrome.",
|
message=f"Unable to extract restock data for this page unfortunately. (Got code {self.fetcher.get_last_status_code()} from server), no embedded stock information was found and nothing interesting in the text, try using this watch with Chrome.",
|
||||||
url=watch.get('url'),
|
url=watch.get('url'),
|
||||||
@@ -229,12 +233,21 @@ class perform_site_check(difference_detection_processor):
|
|||||||
xpath_data=self.fetcher.xpath_data
|
xpath_data=self.fetcher.xpath_data
|
||||||
)
|
)
|
||||||
|
|
||||||
|
logger.debug(f"self.fetcher.instock_data is - '{self.fetcher.instock_data}' and itemprop_availability.get('availability') is {itemprop_availability.get('availability')}")
|
||||||
# Nothing automatic in microdata found, revert to scraping the page
|
# Nothing automatic in microdata found, revert to scraping the page
|
||||||
if self.fetcher.instock_data and itemprop_availability.get('availability') is None:
|
if self.fetcher.instock_data and itemprop_availability.get('availability') is None:
|
||||||
# 'Possibly in stock' comes from stock-not-in-stock.js when no string found above the fold.
|
# 'Possibly in stock' comes from stock-not-in-stock.js when no string found above the fold.
|
||||||
# Careful! this does not really come from chrome/js when the watch is set to plaintext
|
# Careful! this does not really come from chrome/js when the watch is set to plaintext
|
||||||
update_obj['restock']["in_stock"] = True if self.fetcher.instock_data == 'Possibly in stock' else False
|
update_obj['restock']["in_stock"] = True if self.fetcher.instock_data == 'Possibly in stock' else False
|
||||||
logger.debug(f"Watch UUID {watch.get('uuid')} restock check returned '{self.fetcher.instock_data}' from JS scraper.")
|
logger.debug(f"Watch UUID {watch.get('uuid')} restock check returned instock_data - '{self.fetcher.instock_data}' from JS scraper.")
|
||||||
|
|
||||||
|
# Very often websites will lie about the 'availability' in the metadata, so if the scraped version says its NOT in stock, use that.
|
||||||
|
if self.fetcher.instock_data and self.fetcher.instock_data != 'Possibly in stock':
|
||||||
|
if update_obj['restock'].get('in_stock'):
|
||||||
|
logger.warning(
|
||||||
|
f"Lie detected in the availability machine data!! when scraping said its not in stock!! itemprop was '{itemprop_availability}' and scraped from browser was '{self.fetcher.instock_data}' update obj was {update_obj['restock']} ")
|
||||||
|
logger.warning(f"Setting instock to FALSE, scraper found '{self.fetcher.instock_data}' in the body but metadata reported not-in-stock")
|
||||||
|
update_obj['restock']["in_stock"] = False
|
||||||
|
|
||||||
# What we store in the snapshot
|
# What we store in the snapshot
|
||||||
price = update_obj.get('restock').get('price') if update_obj.get('restock').get('price') else ""
|
price = update_obj.get('restock').get('price') if update_obj.get('restock').get('price') else ""
|
||||||
@@ -298,4 +311,4 @@ class perform_site_check(difference_detection_processor):
|
|||||||
# Always record the new checksum
|
# Always record the new checksum
|
||||||
update_obj["previous_md5"] = fetched_md5
|
update_obj["previous_md5"] = fetched_md5
|
||||||
|
|
||||||
return changed_detected, update_obj, snapshot_content.encode('utf-8').strip(), b''
|
return changed_detected, update_obj, snapshot_content.strip()
|
||||||
|
|||||||
@@ -0,0 +1,115 @@
|
|||||||
|
|
||||||
|
from loguru import logger
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def _task(watch, update_handler):
|
||||||
|
from changedetectionio.content_fetchers.exceptions import ReplyWithContentButNoText
|
||||||
|
from changedetectionio.processors.text_json_diff.processor import FilterNotFoundInResponse
|
||||||
|
|
||||||
|
text_after_filter = ''
|
||||||
|
|
||||||
|
try:
|
||||||
|
# The slow process (we run 2 of these in parallel)
|
||||||
|
changed_detected, update_obj, text_after_filter = update_handler.run_changedetection(watch=watch)
|
||||||
|
except FilterNotFoundInResponse as e:
|
||||||
|
text_after_filter = f"Filter not found in HTML: {str(e)}"
|
||||||
|
except ReplyWithContentButNoText as e:
|
||||||
|
text_after_filter = f"Filter found but no text (empty result)"
|
||||||
|
except Exception as e:
|
||||||
|
text_after_filter = f"Error: {str(e)}"
|
||||||
|
|
||||||
|
if not text_after_filter.strip():
|
||||||
|
text_after_filter = 'Empty content'
|
||||||
|
|
||||||
|
# because run_changedetection always returns bytes due to saving the snapshots etc
|
||||||
|
text_after_filter = text_after_filter.decode('utf-8') if isinstance(text_after_filter, bytes) else text_after_filter
|
||||||
|
|
||||||
|
return text_after_filter
|
||||||
|
|
||||||
|
|
||||||
|
def prepare_filter_prevew(datastore, watch_uuid):
|
||||||
|
'''Used by @app.route("/edit/<string:uuid>/preview-rendered", methods=['POST'])'''
|
||||||
|
from changedetectionio import forms, html_tools
|
||||||
|
from changedetectionio.model.Watch import model as watch_model
|
||||||
|
from concurrent.futures import ProcessPoolExecutor
|
||||||
|
from copy import deepcopy
|
||||||
|
from flask import request, jsonify
|
||||||
|
import brotli
|
||||||
|
import importlib
|
||||||
|
import os
|
||||||
|
import time
|
||||||
|
now = time.time()
|
||||||
|
|
||||||
|
text_after_filter = ''
|
||||||
|
text_before_filter = ''
|
||||||
|
trigger_line_numbers = []
|
||||||
|
ignore_line_numbers = []
|
||||||
|
|
||||||
|
tmp_watch = deepcopy(datastore.data['watching'].get(watch_uuid))
|
||||||
|
|
||||||
|
if tmp_watch and tmp_watch.history and os.path.isdir(tmp_watch.watch_data_dir):
|
||||||
|
# Splice in the temporary stuff from the form
|
||||||
|
form = forms.processor_text_json_diff_form(formdata=request.form if request.method == 'POST' else None,
|
||||||
|
data=request.form
|
||||||
|
)
|
||||||
|
|
||||||
|
# Only update vars that came in via the AJAX post
|
||||||
|
p = {k: v for k, v in form.data.items() if k in request.form.keys()}
|
||||||
|
tmp_watch.update(p)
|
||||||
|
blank_watch_no_filters = watch_model()
|
||||||
|
blank_watch_no_filters['url'] = tmp_watch.get('url')
|
||||||
|
|
||||||
|
latest_filename = next(reversed(tmp_watch.history))
|
||||||
|
html_fname = os.path.join(tmp_watch.watch_data_dir, f"{latest_filename}.html.br")
|
||||||
|
with open(html_fname, 'rb') as f:
|
||||||
|
decompressed_data = brotli.decompress(f.read()).decode('utf-8') if html_fname.endswith('.br') else f.read().decode('utf-8')
|
||||||
|
|
||||||
|
# Just like a normal change detection except provide a fake "watch" object and dont call .call_browser()
|
||||||
|
processor_module = importlib.import_module("changedetectionio.processors.text_json_diff.processor")
|
||||||
|
update_handler = processor_module.perform_site_check(datastore=datastore,
|
||||||
|
watch_uuid=tmp_watch.get('uuid') # probably not needed anymore anyway?
|
||||||
|
)
|
||||||
|
# Use the last loaded HTML as the input
|
||||||
|
update_handler.datastore = datastore
|
||||||
|
update_handler.fetcher.content = str(decompressed_data) # str() because playwright/puppeteer/requests return string
|
||||||
|
update_handler.fetcher.headers['content-type'] = tmp_watch.get('content-type')
|
||||||
|
|
||||||
|
# Process our watch with filters and the HTML from disk, and also a blank watch with no filters but also with the same HTML from disk
|
||||||
|
# Do this as a parallel process because it could take some time
|
||||||
|
with ProcessPoolExecutor(max_workers=2) as executor:
|
||||||
|
future1 = executor.submit(_task, tmp_watch, update_handler)
|
||||||
|
future2 = executor.submit(_task, blank_watch_no_filters, update_handler)
|
||||||
|
|
||||||
|
text_after_filter = future1.result()
|
||||||
|
text_before_filter = future2.result()
|
||||||
|
|
||||||
|
try:
|
||||||
|
trigger_line_numbers = html_tools.strip_ignore_text(content=text_after_filter,
|
||||||
|
wordlist=tmp_watch['trigger_text'],
|
||||||
|
mode='line numbers'
|
||||||
|
)
|
||||||
|
except Exception as e:
|
||||||
|
text_before_filter = f"Error: {str(e)}"
|
||||||
|
|
||||||
|
try:
|
||||||
|
text_to_ignore = tmp_watch.get('ignore_text', []) + datastore.data['settings']['application'].get('global_ignore_text', [])
|
||||||
|
ignore_line_numbers = html_tools.strip_ignore_text(content=text_after_filter,
|
||||||
|
wordlist=text_to_ignore,
|
||||||
|
mode='line numbers'
|
||||||
|
)
|
||||||
|
except Exception as e:
|
||||||
|
text_before_filter = f"Error: {str(e)}"
|
||||||
|
|
||||||
|
logger.trace(f"Parsed in {time.time() - now:.3f}s")
|
||||||
|
|
||||||
|
return jsonify(
|
||||||
|
{
|
||||||
|
'after_filter': text_after_filter,
|
||||||
|
'before_filter': text_before_filter.decode('utf-8') if isinstance(text_before_filter, bytes) else text_before_filter,
|
||||||
|
'duration': time.time() - now,
|
||||||
|
'trigger_line_numbers': trigger_line_numbers,
|
||||||
|
'ignore_line_numbers': ignore_line_numbers,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import re
|
|||||||
import urllib3
|
import urllib3
|
||||||
|
|
||||||
from changedetectionio.processors import difference_detection_processor
|
from changedetectionio.processors import difference_detection_processor
|
||||||
from changedetectionio.html_tools import PERL_STYLE_REGEX, cdata_in_document_to_text
|
from changedetectionio.html_tools import PERL_STYLE_REGEX, cdata_in_document_to_text, TRANSLATE_WHITESPACE_TABLE
|
||||||
from changedetectionio import html_tools, content_fetchers
|
from changedetectionio import html_tools, content_fetchers
|
||||||
from changedetectionio.blueprint.price_data_follower import PRICE_DATA_TRACK_ACCEPT, PRICE_DATA_TRACK_REJECT
|
from changedetectionio.blueprint.price_data_follower import PRICE_DATA_TRACK_ACCEPT, PRICE_DATA_TRACK_REJECT
|
||||||
from loguru import logger
|
from loguru import logger
|
||||||
@@ -35,8 +35,7 @@ class PDFToHTMLToolNotFound(ValueError):
|
|||||||
# (set_proxy_from_list)
|
# (set_proxy_from_list)
|
||||||
class perform_site_check(difference_detection_processor):
|
class perform_site_check(difference_detection_processor):
|
||||||
|
|
||||||
def run_changedetection(self, watch, skip_when_checksum_same=True):
|
def run_changedetection(self, watch):
|
||||||
|
|
||||||
changed_detected = False
|
changed_detected = False
|
||||||
html_content = ""
|
html_content = ""
|
||||||
screenshot = False # as bytes
|
screenshot = False # as bytes
|
||||||
@@ -59,9 +58,6 @@ class perform_site_check(difference_detection_processor):
|
|||||||
# Watches added automatically in the queue manager will skip if its the same checksum as the previous run
|
# Watches added automatically in the queue manager will skip if its the same checksum as the previous run
|
||||||
# Saves a lot of CPU
|
# Saves a lot of CPU
|
||||||
update_obj['previous_md5_before_filters'] = hashlib.md5(self.fetcher.content.encode('utf-8')).hexdigest()
|
update_obj['previous_md5_before_filters'] = hashlib.md5(self.fetcher.content.encode('utf-8')).hexdigest()
|
||||||
if skip_when_checksum_same:
|
|
||||||
if update_obj['previous_md5_before_filters'] == watch.get('previous_md5_before_filters'):
|
|
||||||
raise content_fetchers.exceptions.checksumFromPreviousCheckWasTheSame()
|
|
||||||
|
|
||||||
# Fetching complete, now filters
|
# Fetching complete, now filters
|
||||||
|
|
||||||
@@ -202,26 +198,17 @@ class perform_site_check(difference_detection_processor):
|
|||||||
render_anchor_tag_content=do_anchor,
|
render_anchor_tag_content=do_anchor,
|
||||||
is_rss=is_rss) # 1874 activate the <title workaround hack
|
is_rss=is_rss) # 1874 activate the <title workaround hack
|
||||||
|
|
||||||
|
|
||||||
if watch.get('trim_text_whitespace'):
|
if watch.get('trim_text_whitespace'):
|
||||||
stripped_text_from_html = '\n'.join(line.strip() for line in stripped_text_from_html.replace("\n\n", "\n").splitlines())
|
stripped_text_from_html = '\n'.join(line.strip() for line in stripped_text_from_html.replace("\n\n", "\n").splitlines())
|
||||||
|
|
||||||
if watch.get('remove_duplicate_lines'):
|
|
||||||
stripped_text_from_html = '\n'.join(dict.fromkeys(line.strip() for line in stripped_text_from_html.replace("\n\n", "\n").splitlines()))
|
|
||||||
|
|
||||||
if watch.get('sort_text_alphabetically'):
|
|
||||||
# Note: Because a <p>something</p> will add an extra line feed to signify the paragraph gap
|
|
||||||
# we end up with 'Some text\n\n', sorting will add all those extra \n at the start, so we remove them here.
|
|
||||||
stripped_text_from_html = stripped_text_from_html.replace("\n\n", "\n")
|
|
||||||
stripped_text_from_html = '\n'.join(sorted(stripped_text_from_html.splitlines(), key=lambda x: x.lower()))
|
|
||||||
|
|
||||||
|
|
||||||
# Re #340 - return the content before the 'ignore text' was applied
|
# Re #340 - return the content before the 'ignore text' was applied
|
||||||
text_content_before_ignored_filter = stripped_text_from_html.encode('utf-8')
|
# Also used to calculate/show what was removed
|
||||||
|
text_content_before_ignored_filter = stripped_text_from_html
|
||||||
|
|
||||||
# @todo whitespace coming from missing rtrim()?
|
# @todo whitespace coming from missing rtrim()?
|
||||||
# stripped_text_from_html could be based on their preferences, replace the processed text with only that which they want to know about.
|
# stripped_text_from_html could be based on their preferences, replace the processed text with only that which they want to know about.
|
||||||
# Rewrite's the processing text based on only what diff result they want to see
|
# Rewrite's the processing text based on only what diff result they want to see
|
||||||
|
|
||||||
if watch.has_special_diff_filter_options_set() and len(watch.history.keys()):
|
if watch.has_special_diff_filter_options_set() and len(watch.history.keys()):
|
||||||
# Now the content comes from the diff-parser and not the returned HTTP traffic, so could be some differences
|
# Now the content comes from the diff-parser and not the returned HTTP traffic, so could be some differences
|
||||||
from changedetectionio import diff
|
from changedetectionio import diff
|
||||||
@@ -236,13 +223,13 @@ class perform_site_check(difference_detection_processor):
|
|||||||
line_feed_sep="\n",
|
line_feed_sep="\n",
|
||||||
include_change_type_prefix=False)
|
include_change_type_prefix=False)
|
||||||
|
|
||||||
watch.save_last_text_fetched_before_filters(text_content_before_ignored_filter)
|
watch.save_last_text_fetched_before_filters(text_content_before_ignored_filter.encode('utf-8'))
|
||||||
|
|
||||||
if not rendered_diff and stripped_text_from_html:
|
if not rendered_diff and stripped_text_from_html:
|
||||||
# We had some content, but no differences were found
|
# We had some content, but no differences were found
|
||||||
# Store our new file as the MD5 so it will trigger in the future
|
# Store our new file as the MD5 so it will trigger in the future
|
||||||
c = hashlib.md5(text_content_before_ignored_filter.translate(None, b'\r\n\t ')).hexdigest()
|
c = hashlib.md5(stripped_text_from_html.translate(TRANSLATE_WHITESPACE_TABLE).encode('utf-8')).hexdigest()
|
||||||
return False, {'previous_md5': c}, stripped_text_from_html.encode('utf-8'), stripped_text_from_html.encode('utf-8')
|
return False, {'previous_md5': c}, stripped_text_from_html.encode('utf-8')
|
||||||
else:
|
else:
|
||||||
stripped_text_from_html = rendered_diff
|
stripped_text_from_html = rendered_diff
|
||||||
|
|
||||||
@@ -262,14 +249,6 @@ class perform_site_check(difference_detection_processor):
|
|||||||
|
|
||||||
update_obj["last_check_status"] = self.fetcher.get_last_status_code()
|
update_obj["last_check_status"] = self.fetcher.get_last_status_code()
|
||||||
|
|
||||||
# If there's text to skip
|
|
||||||
# @todo we could abstract out the get_text() to handle this cleaner
|
|
||||||
text_to_ignore = watch.get('ignore_text', []) + self.datastore.data['settings']['application'].get('global_ignore_text', [])
|
|
||||||
if len(text_to_ignore):
|
|
||||||
stripped_text_from_html = html_tools.strip_ignore_text(stripped_text_from_html, text_to_ignore)
|
|
||||||
else:
|
|
||||||
stripped_text_from_html = stripped_text_from_html.encode('utf8')
|
|
||||||
|
|
||||||
# 615 Extract text by regex
|
# 615 Extract text by regex
|
||||||
extract_text = watch.get('extract_text', [])
|
extract_text = watch.get('extract_text', [])
|
||||||
if len(extract_text) > 0:
|
if len(extract_text) > 0:
|
||||||
@@ -278,39 +257,53 @@ class perform_site_check(difference_detection_processor):
|
|||||||
# incase they specified something in '/.../x'
|
# incase they specified something in '/.../x'
|
||||||
if re.search(PERL_STYLE_REGEX, s_re, re.IGNORECASE):
|
if re.search(PERL_STYLE_REGEX, s_re, re.IGNORECASE):
|
||||||
regex = html_tools.perl_style_slash_enclosed_regex_to_options(s_re)
|
regex = html_tools.perl_style_slash_enclosed_regex_to_options(s_re)
|
||||||
result = re.findall(regex.encode('utf-8'), stripped_text_from_html)
|
result = re.findall(regex, stripped_text_from_html)
|
||||||
|
|
||||||
for l in result:
|
for l in result:
|
||||||
if type(l) is tuple:
|
if type(l) is tuple:
|
||||||
# @todo - some formatter option default (between groups)
|
# @todo - some formatter option default (between groups)
|
||||||
regex_matched_output += list(l) + [b'\n']
|
regex_matched_output += list(l) + ['\n']
|
||||||
else:
|
else:
|
||||||
# @todo - some formatter option default (between each ungrouped result)
|
# @todo - some formatter option default (between each ungrouped result)
|
||||||
regex_matched_output += [l] + [b'\n']
|
regex_matched_output += [l] + ['\n']
|
||||||
else:
|
else:
|
||||||
# Doesnt look like regex, just hunt for plaintext and return that which matches
|
# Doesnt look like regex, just hunt for plaintext and return that which matches
|
||||||
# `stripped_text_from_html` will be bytes, so we must encode s_re also to bytes
|
# `stripped_text_from_html` will be bytes, so we must encode s_re also to bytes
|
||||||
r = re.compile(re.escape(s_re.encode('utf-8')), re.IGNORECASE)
|
r = re.compile(re.escape(s_re), re.IGNORECASE)
|
||||||
res = r.findall(stripped_text_from_html)
|
res = r.findall(stripped_text_from_html)
|
||||||
if res:
|
if res:
|
||||||
for match in res:
|
for match in res:
|
||||||
regex_matched_output += [match] + [b'\n']
|
regex_matched_output += [match] + ['\n']
|
||||||
|
|
||||||
##########################################################
|
##########################################################
|
||||||
stripped_text_from_html = b''
|
stripped_text_from_html = ''
|
||||||
text_content_before_ignored_filter = b''
|
|
||||||
if regex_matched_output:
|
if regex_matched_output:
|
||||||
# @todo some formatter for presentation?
|
# @todo some formatter for presentation?
|
||||||
stripped_text_from_html = b''.join(regex_matched_output)
|
stripped_text_from_html = ''.join(regex_matched_output)
|
||||||
text_content_before_ignored_filter = stripped_text_from_html
|
|
||||||
|
if watch.get('remove_duplicate_lines'):
|
||||||
|
stripped_text_from_html = '\n'.join(dict.fromkeys(line for line in stripped_text_from_html.replace("\n\n", "\n").splitlines()))
|
||||||
|
|
||||||
|
|
||||||
|
if watch.get('sort_text_alphabetically'):
|
||||||
|
# Note: Because a <p>something</p> will add an extra line feed to signify the paragraph gap
|
||||||
|
# we end up with 'Some text\n\n', sorting will add all those extra \n at the start, so we remove them here.
|
||||||
|
stripped_text_from_html = stripped_text_from_html.replace("\n\n", "\n")
|
||||||
|
stripped_text_from_html = '\n'.join(sorted(stripped_text_from_html.splitlines(), key=lambda x: x.lower()))
|
||||||
|
|
||||||
|
### CALCULATE MD5
|
||||||
|
# If there's text to ignore
|
||||||
|
text_to_ignore = watch.get('ignore_text', []) + self.datastore.data['settings']['application'].get('global_ignore_text', [])
|
||||||
|
text_for_checksuming = stripped_text_from_html
|
||||||
|
if text_to_ignore:
|
||||||
|
text_for_checksuming = html_tools.strip_ignore_text(stripped_text_from_html, text_to_ignore)
|
||||||
|
|
||||||
# Re #133 - if we should strip whitespaces from triggering the change detected comparison
|
# Re #133 - if we should strip whitespaces from triggering the change detected comparison
|
||||||
if self.datastore.data['settings']['application'].get('ignore_whitespace', False):
|
if text_for_checksuming and self.datastore.data['settings']['application'].get('ignore_whitespace', False):
|
||||||
fetched_md5 = hashlib.md5(stripped_text_from_html.translate(None, b'\r\n\t ')).hexdigest()
|
fetched_md5 = hashlib.md5(text_for_checksuming.translate(TRANSLATE_WHITESPACE_TABLE).encode('utf-8')).hexdigest()
|
||||||
else:
|
else:
|
||||||
fetched_md5 = hashlib.md5(stripped_text_from_html).hexdigest()
|
fetched_md5 = hashlib.md5(text_for_checksuming.encode('utf-8')).hexdigest()
|
||||||
|
|
||||||
############ Blocking rules, after checksum #################
|
############ Blocking rules, after checksum #################
|
||||||
blocked = False
|
blocked = False
|
||||||
@@ -338,19 +331,33 @@ class perform_site_check(difference_detection_processor):
|
|||||||
if result:
|
if result:
|
||||||
blocked = True
|
blocked = True
|
||||||
|
|
||||||
# The main thing that all this at the moment comes down to :)
|
|
||||||
if watch.get('previous_md5') != fetched_md5:
|
|
||||||
changed_detected = True
|
|
||||||
|
|
||||||
# Looks like something changed, but did it match all the rules?
|
# Looks like something changed, but did it match all the rules?
|
||||||
if blocked:
|
if blocked:
|
||||||
changed_detected = False
|
changed_detected = False
|
||||||
|
else:
|
||||||
|
# The main thing that all this at the moment comes down to :)
|
||||||
|
if watch.get('previous_md5') != fetched_md5:
|
||||||
|
changed_detected = True
|
||||||
|
|
||||||
|
# Always record the new checksum
|
||||||
|
update_obj["previous_md5"] = fetched_md5
|
||||||
|
|
||||||
|
# On the first run of a site, watch['previous_md5'] will be None, set it the current one.
|
||||||
|
if not watch.get('previous_md5'):
|
||||||
|
watch['previous_md5'] = fetched_md5
|
||||||
|
|
||||||
logger.debug(f"Watch UUID {watch.get('uuid')} content check - Previous MD5: {watch.get('previous_md5')}, Fetched MD5 {fetched_md5}")
|
logger.debug(f"Watch UUID {watch.get('uuid')} content check - Previous MD5: {watch.get('previous_md5')}, Fetched MD5 {fetched_md5}")
|
||||||
|
|
||||||
if changed_detected:
|
if changed_detected:
|
||||||
if watch.get('check_unique_lines', False):
|
if watch.get('check_unique_lines', False):
|
||||||
has_unique_lines = watch.lines_contain_something_unique_compared_to_history(lines=stripped_text_from_html.splitlines())
|
ignore_whitespace = self.datastore.data['settings']['application'].get('ignore_whitespace')
|
||||||
|
|
||||||
|
has_unique_lines = watch.lines_contain_something_unique_compared_to_history(
|
||||||
|
lines=stripped_text_from_html.splitlines(),
|
||||||
|
ignore_whitespace=ignore_whitespace
|
||||||
|
)
|
||||||
|
|
||||||
# One or more lines? unsure?
|
# One or more lines? unsure?
|
||||||
if not has_unique_lines:
|
if not has_unique_lines:
|
||||||
logger.debug(f"check_unique_lines: UUID {watch.get('uuid')} didnt have anything new setting change_detected=False")
|
logger.debug(f"check_unique_lines: UUID {watch.get('uuid')} didnt have anything new setting change_detected=False")
|
||||||
@@ -358,11 +365,6 @@ class perform_site_check(difference_detection_processor):
|
|||||||
else:
|
else:
|
||||||
logger.debug(f"check_unique_lines: UUID {watch.get('uuid')} had unique content")
|
logger.debug(f"check_unique_lines: UUID {watch.get('uuid')} had unique content")
|
||||||
|
|
||||||
# Always record the new checksum
|
|
||||||
update_obj["previous_md5"] = fetched_md5
|
|
||||||
|
|
||||||
# On the first run of a site, watch['previous_md5'] will be None, set it the current one.
|
# stripped_text_from_html - Everything after filters and NO 'ignored' content
|
||||||
if not watch.get('previous_md5'):
|
return changed_detected, update_obj, stripped_text_from_html
|
||||||
watch['previous_md5'] = fetched_md5
|
|
||||||
|
|
||||||
return changed_detected, update_obj, text_content_before_ignored_filter, stripped_text_from_html
|
|
||||||
|
|||||||
@@ -1,56 +0,0 @@
|
|||||||
/**
|
|
||||||
* debounce
|
|
||||||
* @param {integer} milliseconds This param indicates the number of milliseconds
|
|
||||||
* to wait after the last call before calling the original function.
|
|
||||||
* @param {object} What "this" refers to in the returned function.
|
|
||||||
* @return {function} This returns a function that when called will wait the
|
|
||||||
* indicated number of milliseconds after the last call before
|
|
||||||
* calling the original function.
|
|
||||||
*/
|
|
||||||
Function.prototype.debounce = function (milliseconds, context) {
|
|
||||||
var baseFunction = this,
|
|
||||||
timer = null,
|
|
||||||
wait = milliseconds;
|
|
||||||
|
|
||||||
return function () {
|
|
||||||
var self = context || this,
|
|
||||||
args = arguments;
|
|
||||||
|
|
||||||
function complete() {
|
|
||||||
baseFunction.apply(self, args);
|
|
||||||
timer = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (timer) {
|
|
||||||
clearTimeout(timer);
|
|
||||||
}
|
|
||||||
|
|
||||||
timer = setTimeout(complete, wait);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* throttle
|
|
||||||
* @param {integer} milliseconds This param indicates the number of milliseconds
|
|
||||||
* to wait between calls before calling the original function.
|
|
||||||
* @param {object} What "this" refers to in the returned function.
|
|
||||||
* @return {function} This returns a function that when called will wait the
|
|
||||||
* indicated number of milliseconds between calls before
|
|
||||||
* calling the original function.
|
|
||||||
*/
|
|
||||||
Function.prototype.throttle = function (milliseconds, context) {
|
|
||||||
var baseFunction = this,
|
|
||||||
lastEventTimestamp = null,
|
|
||||||
limit = milliseconds;
|
|
||||||
|
|
||||||
return function () {
|
|
||||||
var self = context || this,
|
|
||||||
args = arguments,
|
|
||||||
now = Date.now();
|
|
||||||
|
|
||||||
if (!lastEventTimestamp || now - lastEventTimestamp >= limit) {
|
|
||||||
lastEventTimestamp = now;
|
|
||||||
baseFunction.apply(self, args);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
};
|
|
||||||
162
changedetectionio/static/js/plugins.js
Normal file
162
changedetectionio/static/js/plugins.js
Normal file
@@ -0,0 +1,162 @@
|
|||||||
|
(function ($) {
|
||||||
|
/**
|
||||||
|
* debounce
|
||||||
|
* @param {integer} milliseconds This param indicates the number of milliseconds
|
||||||
|
* to wait after the last call before calling the original function.
|
||||||
|
* @param {object} What "this" refers to in the returned function.
|
||||||
|
* @return {function} This returns a function that when called will wait the
|
||||||
|
* indicated number of milliseconds after the last call before
|
||||||
|
* calling the original function.
|
||||||
|
*/
|
||||||
|
Function.prototype.debounce = function (milliseconds, context) {
|
||||||
|
var baseFunction = this,
|
||||||
|
timer = null,
|
||||||
|
wait = milliseconds;
|
||||||
|
|
||||||
|
return function () {
|
||||||
|
var self = context || this,
|
||||||
|
args = arguments;
|
||||||
|
|
||||||
|
function complete() {
|
||||||
|
baseFunction.apply(self, args);
|
||||||
|
timer = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (timer) {
|
||||||
|
clearTimeout(timer);
|
||||||
|
}
|
||||||
|
|
||||||
|
timer = setTimeout(complete, wait);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* throttle
|
||||||
|
* @param {integer} milliseconds This param indicates the number of milliseconds
|
||||||
|
* to wait between calls before calling the original function.
|
||||||
|
* @param {object} What "this" refers to in the returned function.
|
||||||
|
* @return {function} This returns a function that when called will wait the
|
||||||
|
* indicated number of milliseconds between calls before
|
||||||
|
* calling the original function.
|
||||||
|
*/
|
||||||
|
Function.prototype.throttle = function (milliseconds, context) {
|
||||||
|
var baseFunction = this,
|
||||||
|
lastEventTimestamp = null,
|
||||||
|
limit = milliseconds;
|
||||||
|
|
||||||
|
return function () {
|
||||||
|
var self = context || this,
|
||||||
|
args = arguments,
|
||||||
|
now = Date.now();
|
||||||
|
|
||||||
|
if (!lastEventTimestamp || now - lastEventTimestamp >= limit) {
|
||||||
|
lastEventTimestamp = now;
|
||||||
|
baseFunction.apply(self, args);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
$.fn.highlightLines = function (configurations) {
|
||||||
|
return this.each(function () {
|
||||||
|
const $pre = $(this);
|
||||||
|
const textContent = $pre.text();
|
||||||
|
const lines = textContent.split(/\r?\n/); // Handles both \n and \r\n line endings
|
||||||
|
|
||||||
|
// Build a map of line numbers to styles
|
||||||
|
const lineStyles = {};
|
||||||
|
|
||||||
|
configurations.forEach(config => {
|
||||||
|
const {color, lines: lineNumbers} = config;
|
||||||
|
lineNumbers.forEach(lineNumber => {
|
||||||
|
lineStyles[lineNumber] = color;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Function to escape HTML characters
|
||||||
|
function escapeHtml(text) {
|
||||||
|
return text.replace(/[&<>"'`=\/]/g, function (s) {
|
||||||
|
return "&#" + s.charCodeAt(0) + ";";
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Process each line
|
||||||
|
const processedLines = lines.map((line, index) => {
|
||||||
|
const lineNumber = index + 1; // Line numbers start at 1
|
||||||
|
const escapedLine = escapeHtml(line);
|
||||||
|
const color = lineStyles[lineNumber];
|
||||||
|
|
||||||
|
if (color) {
|
||||||
|
// Wrap the line in a span with inline style
|
||||||
|
return `<span style="background-color: ${color}">${escapedLine}</span>`;
|
||||||
|
} else {
|
||||||
|
return escapedLine;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Join the lines back together
|
||||||
|
const newContent = processedLines.join('\n');
|
||||||
|
|
||||||
|
// Set the new content as HTML
|
||||||
|
$pre.html(newContent);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
$.fn.miniTabs = function (tabsConfig, options) {
|
||||||
|
const settings = {
|
||||||
|
tabClass: 'minitab',
|
||||||
|
tabsContainerClass: 'minitabs',
|
||||||
|
activeClass: 'active',
|
||||||
|
...(options || {})
|
||||||
|
};
|
||||||
|
|
||||||
|
return this.each(function () {
|
||||||
|
const $wrapper = $(this);
|
||||||
|
const $contents = $wrapper.find('div[id]').hide();
|
||||||
|
const $tabsContainer = $('<div>', {class: settings.tabsContainerClass}).prependTo($wrapper);
|
||||||
|
|
||||||
|
// Generate tabs
|
||||||
|
Object.entries(tabsConfig).forEach(([tabTitle, contentSelector], index) => {
|
||||||
|
const $content = $wrapper.find(contentSelector);
|
||||||
|
if (index === 0) $content.show(); // Show first content by default
|
||||||
|
|
||||||
|
$('<a>', {
|
||||||
|
class: `${settings.tabClass}${index === 0 ? ` ${settings.activeClass}` : ''}`,
|
||||||
|
text: tabTitle,
|
||||||
|
'data-target': contentSelector
|
||||||
|
}).appendTo($tabsContainer);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Tab click event
|
||||||
|
$tabsContainer.on('click', `.${settings.tabClass}`, function (e) {
|
||||||
|
e.preventDefault();
|
||||||
|
const $tab = $(this);
|
||||||
|
const target = $tab.data('target');
|
||||||
|
|
||||||
|
// Update active tab
|
||||||
|
$tabsContainer.find(`.${settings.tabClass}`).removeClass(settings.activeClass);
|
||||||
|
$tab.addClass(settings.activeClass);
|
||||||
|
|
||||||
|
// Show/hide content
|
||||||
|
$contents.hide();
|
||||||
|
$wrapper.find(target).show();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// Object to store ongoing requests by namespace
|
||||||
|
const requests = {};
|
||||||
|
|
||||||
|
$.abortiveSingularAjax = function (options) {
|
||||||
|
const namespace = options.namespace || 'default';
|
||||||
|
|
||||||
|
// Abort the current request in this namespace if it's still ongoing
|
||||||
|
if (requests[namespace]) {
|
||||||
|
requests[namespace].abort();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start a new AJAX request and store its reference in the correct namespace
|
||||||
|
requests[namespace] = $.ajax(options);
|
||||||
|
|
||||||
|
// Return the current request in case it's needed
|
||||||
|
return requests[namespace];
|
||||||
|
};
|
||||||
|
})(jQuery);
|
||||||
@@ -1,53 +1,63 @@
|
|||||||
function redirect_to_version(version) {
|
function redirectToVersion(version) {
|
||||||
var currentUrl = window.location.href;
|
var currentUrl = window.location.href.split('?')[0]; // Base URL without query parameters
|
||||||
var baseUrl = currentUrl.split('?')[0]; // Base URL without query parameters
|
|
||||||
var anchor = '';
|
var anchor = '';
|
||||||
|
|
||||||
// Check if there is an anchor
|
// Check if there is an anchor
|
||||||
if (baseUrl.indexOf('#') !== -1) {
|
if (currentUrl.indexOf('#') !== -1) {
|
||||||
anchor = baseUrl.substring(baseUrl.indexOf('#'));
|
anchor = currentUrl.substring(currentUrl.indexOf('#'));
|
||||||
baseUrl = baseUrl.substring(0, baseUrl.indexOf('#'));
|
currentUrl = currentUrl.substring(0, currentUrl.indexOf('#'));
|
||||||
}
|
}
|
||||||
window.location.href = baseUrl + '?version=' + version + anchor;
|
|
||||||
|
window.location.href = currentUrl + '?version=' + version + anchor;
|
||||||
}
|
}
|
||||||
|
|
||||||
document.addEventListener('keydown', function (event) {
|
function setupDateWidget() {
|
||||||
var selectElement = document.getElementById('preview-version');
|
$(document).on('keydown', function (event) {
|
||||||
if (selectElement) {
|
var $selectElement = $('#preview-version');
|
||||||
var selectedOption = selectElement.querySelector('option:checked');
|
var $selectedOption = $selectElement.find('option:selected');
|
||||||
if (selectedOption) {
|
|
||||||
if (event.key === 'ArrowLeft') {
|
if ($selectedOption.length) {
|
||||||
if (selectedOption.previousElementSibling) {
|
if (event.key === 'ArrowLeft' && $selectedOption.prev().length) {
|
||||||
redirect_to_version(selectedOption.previousElementSibling.value);
|
redirectToVersion($selectedOption.prev().val());
|
||||||
}
|
} else if (event.key === 'ArrowRight' && $selectedOption.next().length) {
|
||||||
} else if (event.key === 'ArrowRight') {
|
redirectToVersion($selectedOption.next().val());
|
||||||
if (selectedOption.nextElementSibling) {
|
|
||||||
redirect_to_version(selectedOption.nextElementSibling.value);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
});
|
|
||||||
|
|
||||||
|
$('#preview-version').on('change', function () {
|
||||||
|
redirectToVersion($(this).val());
|
||||||
|
});
|
||||||
|
|
||||||
document.getElementById('preview-version').addEventListener('change', function () {
|
var $selectedOption = $('#preview-version option:selected');
|
||||||
redirect_to_version(this.value);
|
|
||||||
});
|
|
||||||
|
|
||||||
var selectElement = document.getElementById('preview-version');
|
if ($selectedOption.length) {
|
||||||
if (selectElement) {
|
var $prevOption = $selectedOption.prev();
|
||||||
var selectedOption = selectElement.querySelector('option:checked');
|
var $nextOption = $selectedOption.next();
|
||||||
if (selectedOption) {
|
|
||||||
if (selectedOption.previousElementSibling) {
|
if ($prevOption.length) {
|
||||||
document.getElementById('btn-previous').href = "?version=" + selectedOption.previousElementSibling.value;
|
$('#btn-previous').attr('href', '?version=' + $prevOption.val());
|
||||||
} else {
|
} else {
|
||||||
document.getElementById('btn-previous').remove()
|
$('#btn-previous').remove();
|
||||||
}
|
|
||||||
if (selectedOption.nextElementSibling) {
|
|
||||||
document.getElementById('btn-next').href = "?version=" + selectedOption.nextElementSibling.value;
|
|
||||||
} else {
|
|
||||||
document.getElementById('btn-next').remove()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($nextOption.length) {
|
||||||
|
$('#btn-next').attr('href', '?version=' + $nextOption.val());
|
||||||
|
} else {
|
||||||
|
$('#btn-next').remove();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$(document).ready(function () {
|
||||||
|
if ($('#preview-version').length) {
|
||||||
|
setupDateWidget();
|
||||||
|
}
|
||||||
|
|
||||||
|
$('#diff-col > pre').highlightLines([
|
||||||
|
{
|
||||||
|
'color': '#ee0000',
|
||||||
|
'lines': triggered_line_numbers
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
$(function () {
|
$(function () {
|
||||||
/* add container before each proxy location to show status */
|
/* add container before each proxy location to show status */
|
||||||
|
|
||||||
var option_li = $('.fetch-backend-proxy li').filter(function() {
|
|
||||||
return $("input",this)[0].value.length >0;
|
|
||||||
});
|
|
||||||
|
|
||||||
//var option_li = $('.fetch-backend-proxy li');
|
|
||||||
var isActive = false;
|
var isActive = false;
|
||||||
$(option_li).prepend('<div class="proxy-status"></div>');
|
|
||||||
$(option_li).append('<div class="proxy-timing"></div><div class="proxy-check-details"></div>');
|
function setup_html_widget() {
|
||||||
|
var option_li = $('.fetch-backend-proxy li').filter(function () {
|
||||||
|
return $("input", this)[0].value.length > 0;
|
||||||
|
});
|
||||||
|
$(option_li).prepend('<div class="proxy-status"></div>');
|
||||||
|
$(option_li).append('<div class="proxy-timing"></div><div class="proxy-check-details"></div>');
|
||||||
|
}
|
||||||
|
|
||||||
function set_proxy_check_status(proxy_key, state) {
|
function set_proxy_check_status(proxy_key, state) {
|
||||||
// select input by value name
|
// select input by value name
|
||||||
@@ -59,8 +59,14 @@ $(function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
$('#check-all-proxies').click(function (e) {
|
$('#check-all-proxies').click(function (e) {
|
||||||
|
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
$('body').addClass('proxy-check-active');
|
|
||||||
|
if (!$('body').hasClass('proxy-check-active')) {
|
||||||
|
setup_html_widget();
|
||||||
|
$('body').addClass('proxy-check-active');
|
||||||
|
}
|
||||||
|
|
||||||
$('.proxy-check-details').html('');
|
$('.proxy-check-details').html('');
|
||||||
$('.proxy-status').html('<span class="spinner"></span>').fadeIn();
|
$('.proxy-status').html('<span class="spinner"></span>').fadeIn();
|
||||||
$('.proxy-timing').html('');
|
$('.proxy-timing').html('');
|
||||||
|
|||||||
@@ -26,8 +26,7 @@ function set_active_tab() {
|
|||||||
if (tab.length) {
|
if (tab.length) {
|
||||||
tab[0].parentElement.className = "active";
|
tab[0].parentElement.className = "active";
|
||||||
}
|
}
|
||||||
// hash could move the page down
|
|
||||||
window.scrollTo(0, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function focus_error_tab() {
|
function focus_error_tab() {
|
||||||
|
|||||||
@@ -49,4 +49,9 @@ $(document).ready(function () {
|
|||||||
$("#overlay").toggleClass('visible');
|
$("#overlay").toggleClass('visible');
|
||||||
heartpath.style.fill = document.getElementById("overlay").classList.contains("visible") ? '#ff0000' : 'var(--color-background)';
|
heartpath.style.fill = document.getElementById("overlay").classList.contains("visible") ? '#ff0000' : 'var(--color-background)';
|
||||||
});
|
});
|
||||||
|
|
||||||
|
setInterval(function () {
|
||||||
|
$('body').toggleClass('spinner-active', $.active > 0);
|
||||||
|
}, 2000);
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -12,28 +12,10 @@ function toggleOpacity(checkboxSelector, fieldSelector, inverted) {
|
|||||||
checkbox.addEventListener('change', updateOpacity);
|
checkbox.addEventListener('change', updateOpacity);
|
||||||
}
|
}
|
||||||
|
|
||||||
(function($) {
|
|
||||||
// Object to store ongoing requests by namespace
|
|
||||||
const requests = {};
|
|
||||||
|
|
||||||
$.abortiveSingularAjax = function(options) {
|
|
||||||
const namespace = options.namespace || 'default';
|
|
||||||
|
|
||||||
// Abort the current request in this namespace if it's still ongoing
|
|
||||||
if (requests[namespace]) {
|
|
||||||
requests[namespace].abort();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start a new AJAX request and store its reference in the correct namespace
|
|
||||||
requests[namespace] = $.ajax(options);
|
|
||||||
|
|
||||||
// Return the current request in case it's needed
|
|
||||||
return requests[namespace];
|
|
||||||
};
|
|
||||||
})(jQuery);
|
|
||||||
|
|
||||||
function request_textpreview_update() {
|
function request_textpreview_update() {
|
||||||
if (!$('body').hasClass('preview-text-enabled')) {
|
if (!$('body').hasClass('preview-text-enabled')) {
|
||||||
|
console.error("Preview text was requested but body tag was not setup")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -41,16 +23,31 @@ function request_textpreview_update() {
|
|||||||
$('textarea:visible, input:visible').each(function () {
|
$('textarea:visible, input:visible').each(function () {
|
||||||
const $element = $(this); // Cache the jQuery object for the current element
|
const $element = $(this); // Cache the jQuery object for the current element
|
||||||
const name = $element.attr('name'); // Get the name attribute of the element
|
const name = $element.attr('name'); // Get the name attribute of the element
|
||||||
data[name] = $element.is(':checkbox') ? ($element.is(':checked') ? $element.val() : undefined) : $element.val();
|
data[name] = $element.is(':checkbox') ? ($element.is(':checked') ? $element.val() : false) : $element.val();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
$('body').toggleClass('spinner-active', 1);
|
||||||
|
|
||||||
$.abortiveSingularAjax({
|
$.abortiveSingularAjax({
|
||||||
type: "POST",
|
type: "POST",
|
||||||
url: preview_text_edit_filters_url,
|
url: preview_text_edit_filters_url,
|
||||||
data: data,
|
data: data,
|
||||||
namespace: 'watchEdit'
|
namespace: 'watchEdit'
|
||||||
}).done(function (data) {
|
}).done(function (data) {
|
||||||
$('#filters-and-triggers #text-preview-inner').text(data);
|
console.debug(data['duration'])
|
||||||
|
$('#filters-and-triggers #text-preview-before-inner').text(data['before_filter']);
|
||||||
|
$('#filters-and-triggers #text-preview-inner')
|
||||||
|
.text(data['after_filter'])
|
||||||
|
.highlightLines([
|
||||||
|
{
|
||||||
|
'color': '#ee0000',
|
||||||
|
'lines': data['trigger_line_numbers']
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'color': '#757575',
|
||||||
|
'lines': data['ignore_line_numbers']
|
||||||
|
}
|
||||||
|
])
|
||||||
}).fail(function (error) {
|
}).fail(function (error) {
|
||||||
if (error.statusText === 'abort') {
|
if (error.statusText === 'abort') {
|
||||||
console.log('Request was aborted due to a new request being fired.');
|
console.log('Request was aborted due to a new request being fired.');
|
||||||
@@ -77,21 +74,19 @@ $(document).ready(function () {
|
|||||||
|
|
||||||
const vh = Math.max(document.documentElement.clientHeight || 0, window.innerHeight || 0);
|
const vh = Math.max(document.documentElement.clientHeight || 0, window.innerHeight || 0);
|
||||||
$("#text-preview-inner").css('max-height', (vh-300)+"px");
|
$("#text-preview-inner").css('max-height', (vh-300)+"px");
|
||||||
var debounced_request_textpreview_update = request_textpreview_update.debounce(100);
|
$("#text-preview-before-inner").css('max-height', (vh-300)+"px");
|
||||||
|
|
||||||
$("#activate-text-preview").click(function (e) {
|
$("#activate-text-preview").click(function (e) {
|
||||||
$(this).fadeOut();
|
|
||||||
$('body').toggleClass('preview-text-enabled')
|
$('body').toggleClass('preview-text-enabled')
|
||||||
|
|
||||||
request_textpreview_update();
|
request_textpreview_update();
|
||||||
|
const method = $('body').hasClass('preview-text-enabled') ? 'on' : 'off';
|
||||||
$("#text-preview-refresh").click(function (e) {
|
$('#filters-and-triggers textarea')[method]('blur', request_textpreview_update.throttle(1000));
|
||||||
request_textpreview_update();
|
$('#filters-and-triggers input')[method]('change', request_textpreview_update.throttle(1000));
|
||||||
});
|
$("#filters-and-triggers-tab")[method]('click', request_textpreview_update.throttle(1000));
|
||||||
$('textarea:visible').on('keyup blur', debounced_request_textpreview_update);
|
});
|
||||||
$('input:visible').on('keyup blur change', debounced_request_textpreview_update);
|
$('.minitabs-wrapper').miniTabs({
|
||||||
$("#filters-and-triggers-tab").on('click', debounced_request_textpreview_update);
|
"Content after filters": "#text-preview-inner",
|
||||||
|
"Content raw/before filters": "#text-preview-before-inner"
|
||||||
});
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -25,15 +25,19 @@ ul#requests-extra_proxies {
|
|||||||
|
|
||||||
body.proxy-check-active {
|
body.proxy-check-active {
|
||||||
#request {
|
#request {
|
||||||
|
// Padding set by flex layout
|
||||||
|
/*
|
||||||
.proxy-status {
|
.proxy-status {
|
||||||
width: 2em;
|
width: 2em;
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
.proxy-check-details {
|
.proxy-check-details {
|
||||||
font-size: 80%;
|
font-size: 80%;
|
||||||
color: #555;
|
color: #555;
|
||||||
display: block;
|
display: block;
|
||||||
padding-left: 4em;
|
padding-left: 2em;
|
||||||
|
max-width: 500px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.proxy-timing {
|
.proxy-timing {
|
||||||
|
|||||||
47
changedetectionio/static/styles/scss/parts/_minitabs.scss
Normal file
47
changedetectionio/static/styles/scss/parts/_minitabs.scss
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
.minitabs-wrapper {
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
> div[id] {
|
||||||
|
padding: 20px;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-top: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.minitabs-content {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
> div {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
min-width: 0;
|
||||||
|
overflow: scroll;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.minitabs {
|
||||||
|
display: flex;
|
||||||
|
border-bottom: 1px solid #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
.minitab {
|
||||||
|
flex: 1;
|
||||||
|
text-align: center;
|
||||||
|
padding: 12px 0;
|
||||||
|
text-decoration: none;
|
||||||
|
color: #333;
|
||||||
|
background-color: #f1f1f1;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-bottom: none;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.minitab:hover {
|
||||||
|
background-color: #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
.minitab.active {
|
||||||
|
background-color: #fff;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -1,8 +1,13 @@
|
|||||||
|
@import "minitabs";
|
||||||
|
|
||||||
body.preview-text-enabled {
|
body.preview-text-enabled {
|
||||||
#filters-and-triggers > div {
|
|
||||||
display: flex; /* Establishes Flexbox layout */
|
@media (min-width: 800px) {
|
||||||
gap: 20px; /* Adds space between the columns */
|
#filters-and-triggers > div {
|
||||||
position: relative; /* Ensures the sticky positioning is relative to this parent */
|
display: flex; /* Establishes Flexbox layout */
|
||||||
|
gap: 20px; /* Adds space between the columns */
|
||||||
|
position: relative; /* Ensures the sticky positioning is relative to this parent */
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* layout of the page */
|
/* layout of the page */
|
||||||
@@ -19,27 +24,32 @@ body.preview-text-enabled {
|
|||||||
|
|
||||||
#text-preview {
|
#text-preview {
|
||||||
position: sticky;
|
position: sticky;
|
||||||
top: 25px;
|
top: 20px;
|
||||||
|
padding-top: 1rem;
|
||||||
|
padding-bottom: 1rem;
|
||||||
display: block !important;
|
display: block !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#activate-text-preview {
|
||||||
|
background-color: var(--color-grey-500);
|
||||||
|
}
|
||||||
|
|
||||||
/* actual preview area */
|
/* actual preview area */
|
||||||
#text-preview-inner {
|
.monospace-preview {
|
||||||
background: var(--color-grey-900);
|
background: var(--color-background-input);
|
||||||
border: 1px solid var(--color-grey-600);
|
border: 1px solid var(--color-grey-600);
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
color: #333;
|
color: var(--color-text-input);
|
||||||
font-family: "Courier New", Courier, monospace; /* Sets the font to a monospace type */
|
font-family: "Courier New", Courier, monospace; /* Sets the font to a monospace type */
|
||||||
font-size: 12px;
|
font-size: 70%;
|
||||||
overflow-x: scroll;
|
word-break: break-word;
|
||||||
white-space: pre-wrap; /* Preserves whitespace and line breaks like <pre> */
|
white-space: pre-wrap; /* Preserves whitespace and line breaks like <pre> */
|
||||||
overflow-wrap: break-word; /* Allows long words to break and wrap to the next line */
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#activate-text-preview {
|
#activate-text-preview {
|
||||||
right: 0;
|
right: 0;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
z-index: 0;
|
z-index: 3;
|
||||||
box-shadow: 1px 1px 4px var(--color-shadow-jump);
|
box-shadow: 1px 1px 4px var(--color-shadow-jump);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -106,10 +106,34 @@ button.toggle-button {
|
|||||||
padding: 5px;
|
padding: 5px;
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
border-bottom: 2px solid var(--color-menu-accent);
|
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#pure-menu-horizontal-spinner {
|
||||||
|
height: 3px;
|
||||||
|
background: linear-gradient(-75deg, #ff6000, #ff8f00, #ffdd00, #ed0000);
|
||||||
|
background-size: 400% 400%;
|
||||||
|
width: 100%;
|
||||||
|
animation: gradient 200s ease infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
body.spinner-active {
|
||||||
|
#pure-menu-horizontal-spinner {
|
||||||
|
animation: gradient 1s ease infinite;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes gradient {
|
||||||
|
0% {
|
||||||
|
background-position: 0% 50%;
|
||||||
|
}
|
||||||
|
50% {
|
||||||
|
background-position: 100% 50%;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
background-position: 0% 50%;
|
||||||
|
}
|
||||||
|
}
|
||||||
.pure-menu-heading {
|
.pure-menu-heading {
|
||||||
color: var(--color-text-menu-heading);
|
color: var(--color-text-menu-heading);
|
||||||
}
|
}
|
||||||
@@ -123,8 +147,14 @@ button.toggle-button {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.tab-pane-inner {
|
||||||
|
// .tab-pane-inner will have the #id that the tab button jumps/anchors to
|
||||||
|
scroll-margin-top: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
section.content {
|
section.content {
|
||||||
padding-top: 5em;
|
padding-top: 100px;
|
||||||
padding-bottom: 1em;
|
padding-bottom: 1em;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -321,10 +351,6 @@ a.pure-button-selected {
|
|||||||
background: var(--color-background-button-cancel);
|
background: var(--color-background-button-cancel);
|
||||||
}
|
}
|
||||||
|
|
||||||
#save_button {
|
|
||||||
margin-right: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.messages {
|
.messages {
|
||||||
li {
|
li {
|
||||||
list-style: none;
|
list-style: none;
|
||||||
@@ -621,9 +647,9 @@ footer {
|
|||||||
list-style: none;
|
list-style: none;
|
||||||
|
|
||||||
li {
|
li {
|
||||||
>* {
|
display: flex;
|
||||||
display: inline-block;
|
align-items: center;
|
||||||
}
|
gap: 1em;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -683,6 +709,12 @@ footer {
|
|||||||
tr {
|
tr {
|
||||||
th {
|
th {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
|
// Hide the "Last" text for smaller screens
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.hide-on-mobile {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.empty-cell {
|
.empty-cell {
|
||||||
@@ -698,6 +730,24 @@ footer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tbody {
|
||||||
|
tr {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
|
||||||
|
// The third child of each row will take up the remaining space
|
||||||
|
// This is useful for the URL column, which should expand to fill the remaining space
|
||||||
|
:nth-child(3) {
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
// The last three children (from the end) of each row will take up the full width
|
||||||
|
// This is useful for the "Last Checked", "Last Changed", and the action buttons columns, which should each take up the full width
|
||||||
|
:nth-last-child(-n+3) {
|
||||||
|
flex-basis: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.last-checked {
|
.last-checked {
|
||||||
>span {
|
>span {
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
@@ -816,6 +866,11 @@ textarea::placeholder {
|
|||||||
- We dont use 'size' with <input> because `size` is too unreliable to override, and will often push-out
|
- We dont use 'size' with <input> because `size` is too unreliable to override, and will often push-out
|
||||||
- Rely always on width in CSS
|
- Rely always on width in CSS
|
||||||
*/
|
*/
|
||||||
|
/** Set max width for input field */
|
||||||
|
.m-d {
|
||||||
|
min-width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
@media only screen and (min-width: 761px) {
|
@media only screen and (min-width: 761px) {
|
||||||
|
|
||||||
/* m-d is medium-desktop */
|
/* m-d is medium-desktop */
|
||||||
@@ -882,6 +937,7 @@ $form-edge-padding: 20px;
|
|||||||
}
|
}
|
||||||
|
|
||||||
.tab-pane-inner {
|
.tab-pane-inner {
|
||||||
|
|
||||||
&:not(:target) {
|
&:not(:target) {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
@@ -931,6 +987,13 @@ body.full-width {
|
|||||||
background: var(--color-background);
|
background: var(--color-background);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Make action buttons have consistent size and spacing */
|
||||||
|
#actions .pure-control-group {
|
||||||
|
display: flex;
|
||||||
|
gap: 0.625em;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
.pure-form-message-inline {
|
.pure-form-message-inline {
|
||||||
padding-left: 0;
|
padding-left: 0;
|
||||||
color: var(--color-text-input-description);
|
color: var(--color-text-input-description);
|
||||||
@@ -974,6 +1037,28 @@ ul {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@media only screen and (max-width: 760px) {
|
||||||
|
.time-check-widget {
|
||||||
|
tbody {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: auto 1fr auto 1fr;
|
||||||
|
gap: 0.625em 0.3125em;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
display: contents;
|
||||||
|
th {
|
||||||
|
text-align: right;
|
||||||
|
padding-right: 5px;
|
||||||
|
}
|
||||||
|
input[type="number"] {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 5em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@import "parts/_visualselector";
|
@import "parts/_visualselector";
|
||||||
|
|
||||||
#webdriver_delay {
|
#webdriver_delay {
|
||||||
|
|||||||
@@ -119,19 +119,22 @@ ul#requests-extra_proxies {
|
|||||||
#request label[for=proxy] {
|
#request label[for=proxy] {
|
||||||
display: inline-block; }
|
display: inline-block; }
|
||||||
|
|
||||||
body.proxy-check-active #request .proxy-status {
|
body.proxy-check-active #request {
|
||||||
width: 2em; }
|
/*
|
||||||
|
.proxy-status {
|
||||||
body.proxy-check-active #request .proxy-check-details {
|
width: 2em;
|
||||||
font-size: 80%;
|
}
|
||||||
color: #555;
|
*/ }
|
||||||
display: block;
|
body.proxy-check-active #request .proxy-check-details {
|
||||||
padding-left: 4em; }
|
font-size: 80%;
|
||||||
|
color: #555;
|
||||||
body.proxy-check-active #request .proxy-timing {
|
display: block;
|
||||||
font-size: 80%;
|
padding-left: 2em;
|
||||||
padding-left: 1rem;
|
max-width: 500px; }
|
||||||
color: var(--color-link); }
|
body.proxy-check-active #request .proxy-timing {
|
||||||
|
font-size: 80%;
|
||||||
|
padding-left: 1rem;
|
||||||
|
color: var(--color-link); }
|
||||||
|
|
||||||
#recommended-proxy {
|
#recommended-proxy {
|
||||||
display: grid;
|
display: grid;
|
||||||
@@ -428,16 +431,50 @@ html[data-darkmode="true"] #toggle-light-mode .icon-dark {
|
|||||||
fill: #ff0000 !important;
|
fill: #ff0000 !important;
|
||||||
transition: all ease 0.3s !important; }
|
transition: all ease 0.3s !important; }
|
||||||
|
|
||||||
|
.minitabs-wrapper {
|
||||||
|
width: 100%; }
|
||||||
|
.minitabs-wrapper > div[id] {
|
||||||
|
padding: 20px;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-top: none; }
|
||||||
|
.minitabs-wrapper .minitabs-content {
|
||||||
|
width: 100%;
|
||||||
|
display: flex; }
|
||||||
|
.minitabs-wrapper .minitabs-content > div {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
min-width: 0;
|
||||||
|
overflow: scroll; }
|
||||||
|
.minitabs-wrapper .minitabs {
|
||||||
|
display: flex;
|
||||||
|
border-bottom: 1px solid #ccc; }
|
||||||
|
.minitabs-wrapper .minitab {
|
||||||
|
flex: 1;
|
||||||
|
text-align: center;
|
||||||
|
padding: 12px 0;
|
||||||
|
text-decoration: none;
|
||||||
|
color: #333;
|
||||||
|
background-color: #f1f1f1;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-bottom: none;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.3s; }
|
||||||
|
.minitabs-wrapper .minitab:hover {
|
||||||
|
background-color: #ddd; }
|
||||||
|
.minitabs-wrapper .minitab.active {
|
||||||
|
background-color: #fff;
|
||||||
|
font-weight: bold; }
|
||||||
|
|
||||||
body.preview-text-enabled {
|
body.preview-text-enabled {
|
||||||
/* layout of the page */
|
/* layout of the page */
|
||||||
/* actual preview area */ }
|
/* actual preview area */ }
|
||||||
body.preview-text-enabled #filters-and-triggers > div {
|
@media (min-width: 800px) {
|
||||||
display: flex;
|
body.preview-text-enabled #filters-and-triggers > div {
|
||||||
/* Establishes Flexbox layout */
|
display: flex;
|
||||||
gap: 20px;
|
/* Establishes Flexbox layout */
|
||||||
/* Adds space between the columns */
|
gap: 20px;
|
||||||
position: relative;
|
/* Adds space between the columns */
|
||||||
/* Ensures the sticky positioning is relative to this parent */ }
|
position: relative;
|
||||||
|
/* Ensures the sticky positioning is relative to this parent */ } }
|
||||||
body.preview-text-enabled #edit-text-filter, body.preview-text-enabled #text-preview {
|
body.preview-text-enabled #edit-text-filter, body.preview-text-enabled #text-preview {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
/* Each column takes an equal amount of available space */
|
/* Each column takes an equal amount of available space */
|
||||||
@@ -447,26 +484,28 @@ body.preview-text-enabled {
|
|||||||
display: none; }
|
display: none; }
|
||||||
body.preview-text-enabled #text-preview {
|
body.preview-text-enabled #text-preview {
|
||||||
position: sticky;
|
position: sticky;
|
||||||
top: 25px;
|
top: 20px;
|
||||||
|
padding-top: 1rem;
|
||||||
|
padding-bottom: 1rem;
|
||||||
display: block !important; }
|
display: block !important; }
|
||||||
body.preview-text-enabled #text-preview-inner {
|
body.preview-text-enabled #activate-text-preview {
|
||||||
background: var(--color-grey-900);
|
background-color: var(--color-grey-500); }
|
||||||
|
body.preview-text-enabled .monospace-preview {
|
||||||
|
background: var(--color-background-input);
|
||||||
border: 1px solid var(--color-grey-600);
|
border: 1px solid var(--color-grey-600);
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
color: #333;
|
color: var(--color-text-input);
|
||||||
font-family: "Courier New", Courier, monospace;
|
font-family: "Courier New", Courier, monospace;
|
||||||
/* Sets the font to a monospace type */
|
/* Sets the font to a monospace type */
|
||||||
font-size: 12px;
|
font-size: 70%;
|
||||||
overflow-x: scroll;
|
word-break: break-word;
|
||||||
white-space: pre-wrap;
|
white-space: pre-wrap;
|
||||||
/* Preserves whitespace and line breaks like <pre> */
|
/* Preserves whitespace and line breaks like <pre> */ }
|
||||||
overflow-wrap: break-word;
|
|
||||||
/* Allows long words to break and wrap to the next line */ }
|
|
||||||
|
|
||||||
#activate-text-preview {
|
#activate-text-preview {
|
||||||
right: 0;
|
right: 0;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
z-index: 0;
|
z-index: 3;
|
||||||
box-shadow: 1px 1px 4px var(--color-shadow-jump); }
|
box-shadow: 1px 1px 4px var(--color-shadow-jump); }
|
||||||
|
|
||||||
body {
|
body {
|
||||||
@@ -537,9 +576,26 @@ button.toggle-button {
|
|||||||
padding: 5px;
|
padding: 5px;
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
border-bottom: 2px solid var(--color-menu-accent);
|
|
||||||
align-items: center; }
|
align-items: center; }
|
||||||
|
|
||||||
|
#pure-menu-horizontal-spinner {
|
||||||
|
height: 3px;
|
||||||
|
background: linear-gradient(-75deg, #ff6000, #ff8f00, #ffdd00, #ed0000);
|
||||||
|
background-size: 400% 400%;
|
||||||
|
width: 100%;
|
||||||
|
animation: gradient 200s ease infinite; }
|
||||||
|
|
||||||
|
body.spinner-active #pure-menu-horizontal-spinner {
|
||||||
|
animation: gradient 1s ease infinite; }
|
||||||
|
|
||||||
|
@keyframes gradient {
|
||||||
|
0% {
|
||||||
|
background-position: 0% 50%; }
|
||||||
|
50% {
|
||||||
|
background-position: 100% 50%; }
|
||||||
|
100% {
|
||||||
|
background-position: 0% 50%; } }
|
||||||
|
|
||||||
.pure-menu-heading {
|
.pure-menu-heading {
|
||||||
color: var(--color-text-menu-heading); }
|
color: var(--color-text-menu-heading); }
|
||||||
|
|
||||||
@@ -549,8 +605,11 @@ button.toggle-button {
|
|||||||
background-color: var(--color-background-menu-link-hover);
|
background-color: var(--color-background-menu-link-hover);
|
||||||
color: var(--color-text-menu-link-hover); }
|
color: var(--color-text-menu-link-hover); }
|
||||||
|
|
||||||
|
.tab-pane-inner {
|
||||||
|
scroll-margin-top: 200px; }
|
||||||
|
|
||||||
section.content {
|
section.content {
|
||||||
padding-top: 5em;
|
padding-top: 100px;
|
||||||
padding-bottom: 1em;
|
padding-bottom: 1em;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -692,9 +751,6 @@ a.pure-button-selected {
|
|||||||
.button-cancel {
|
.button-cancel {
|
||||||
background: var(--color-background-button-cancel); }
|
background: var(--color-background-button-cancel); }
|
||||||
|
|
||||||
#save_button {
|
|
||||||
margin-right: 1rem; }
|
|
||||||
|
|
||||||
.messages li {
|
.messages li {
|
||||||
list-style: none;
|
list-style: none;
|
||||||
padding: 1em;
|
padding: 1em;
|
||||||
@@ -893,8 +949,10 @@ footer {
|
|||||||
.pure-form .inline-radio ul {
|
.pure-form .inline-radio ul {
|
||||||
margin: 0px;
|
margin: 0px;
|
||||||
list-style: none; }
|
list-style: none; }
|
||||||
.pure-form .inline-radio ul li > * {
|
.pure-form .inline-radio ul li {
|
||||||
display: inline-block; }
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 1em; }
|
||||||
|
|
||||||
@media only screen and (max-width: 760px), (min-device-width: 768px) and (max-device-width: 1024px) {
|
@media only screen and (max-width: 760px), (min-device-width: 768px) and (max-device-width: 1024px) {
|
||||||
.box {
|
.box {
|
||||||
@@ -930,12 +988,24 @@ footer {
|
|||||||
.watch-table thead {
|
.watch-table thead {
|
||||||
display: block; }
|
display: block; }
|
||||||
.watch-table thead tr th {
|
.watch-table thead tr th {
|
||||||
display: inline-block; }
|
display: inline-block; } }
|
||||||
|
@media only screen and (max-width: 760px) and (max-width: 768px), (min-device-width: 768px) and (max-device-width: 800px) and (max-width: 768px) {
|
||||||
|
.watch-table thead tr th .hide-on-mobile {
|
||||||
|
display: none; } }
|
||||||
|
|
||||||
|
@media only screen and (max-width: 760px), (min-device-width: 768px) and (max-device-width: 800px) {
|
||||||
.watch-table thead .empty-cell {
|
.watch-table thead .empty-cell {
|
||||||
display: none; }
|
display: none; }
|
||||||
.watch-table tbody td,
|
.watch-table tbody td,
|
||||||
.watch-table tbody tr {
|
.watch-table tbody tr {
|
||||||
display: block; }
|
display: block; }
|
||||||
|
.watch-table tbody tr {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap; }
|
||||||
|
.watch-table tbody tr :nth-child(3) {
|
||||||
|
flex-grow: 1; }
|
||||||
|
.watch-table tbody tr :nth-last-child(-n+3) {
|
||||||
|
flex-basis: 100%; }
|
||||||
.watch-table .last-checked > span {
|
.watch-table .last-checked > span {
|
||||||
vertical-align: middle; }
|
vertical-align: middle; }
|
||||||
.watch-table .last-checked::before {
|
.watch-table .last-checked::before {
|
||||||
@@ -1027,6 +1097,10 @@ textarea::placeholder {
|
|||||||
- We dont use 'size' with <input> because `size` is too unreliable to override, and will often push-out
|
- We dont use 'size' with <input> because `size` is too unreliable to override, and will often push-out
|
||||||
- Rely always on width in CSS
|
- Rely always on width in CSS
|
||||||
*/
|
*/
|
||||||
|
/** Set max width for input field */
|
||||||
|
.m-d {
|
||||||
|
min-width: 100%; }
|
||||||
|
|
||||||
@media only screen and (min-width: 761px) {
|
@media only screen and (min-width: 761px) {
|
||||||
/* m-d is medium-desktop */
|
/* m-d is medium-desktop */
|
||||||
.m-d {
|
.m-d {
|
||||||
@@ -1087,7 +1161,8 @@ body.full-width .edit-form {
|
|||||||
.edit-form {
|
.edit-form {
|
||||||
min-width: 70%;
|
min-width: 70%;
|
||||||
/* so it cant overflow */
|
/* so it cant overflow */
|
||||||
max-width: 95%; }
|
max-width: 95%;
|
||||||
|
/* Make action buttons have consistent size and spacing */ }
|
||||||
.edit-form .box-wrap {
|
.edit-form .box-wrap {
|
||||||
position: relative; }
|
position: relative; }
|
||||||
.edit-form .inner {
|
.edit-form .inner {
|
||||||
@@ -1096,6 +1171,10 @@ body.full-width .edit-form {
|
|||||||
.edit-form #actions {
|
.edit-form #actions {
|
||||||
display: block;
|
display: block;
|
||||||
background: var(--color-background); }
|
background: var(--color-background); }
|
||||||
|
.edit-form #actions .pure-control-group {
|
||||||
|
display: flex;
|
||||||
|
gap: 0.625em;
|
||||||
|
flex-wrap: wrap; }
|
||||||
.edit-form .pure-form-message-inline {
|
.edit-form .pure-form-message-inline {
|
||||||
padding-left: 0;
|
padding-left: 0;
|
||||||
color: var(--color-text-input-description); }
|
color: var(--color-text-input-description); }
|
||||||
@@ -1124,6 +1203,21 @@ ul {
|
|||||||
.time-check-widget tr input[type="number"] {
|
.time-check-widget tr input[type="number"] {
|
||||||
width: 5em; }
|
width: 5em; }
|
||||||
|
|
||||||
|
@media only screen and (max-width: 760px) {
|
||||||
|
.time-check-widget tbody {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: auto 1fr auto 1fr;
|
||||||
|
gap: 0.625em 0.3125em;
|
||||||
|
align-items: center; }
|
||||||
|
.time-check-widget tr {
|
||||||
|
display: contents; }
|
||||||
|
.time-check-widget tr th {
|
||||||
|
text-align: right;
|
||||||
|
padding-right: 5px; }
|
||||||
|
.time-check-widget tr input[type="number"] {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 5em; } }
|
||||||
|
|
||||||
#selector-wrapper {
|
#selector-wrapper {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ from flask import (
|
|||||||
flash
|
flash
|
||||||
)
|
)
|
||||||
|
|
||||||
|
from .html_tools import TRANSLATE_WHITESPACE_TABLE
|
||||||
from . model import App, Watch
|
from . model import App, Watch
|
||||||
from copy import deepcopy, copy
|
from copy import deepcopy, copy
|
||||||
from os import path, unlink
|
from os import path, unlink
|
||||||
@@ -750,17 +751,17 @@ class ChangeDetectionStore:
|
|||||||
def update_5(self):
|
def update_5(self):
|
||||||
# If the watch notification body, title look the same as the global one, unset it, so the watch defaults back to using the main settings
|
# If the watch notification body, title look the same as the global one, unset it, so the watch defaults back to using the main settings
|
||||||
# In other words - the watch notification_title and notification_body are not needed if they are the same as the default one
|
# In other words - the watch notification_title and notification_body are not needed if they are the same as the default one
|
||||||
current_system_body = self.data['settings']['application']['notification_body'].translate(str.maketrans('', '', "\r\n "))
|
current_system_body = self.data['settings']['application']['notification_body'].translate(TRANSLATE_WHITESPACE_TABLE)
|
||||||
current_system_title = self.data['settings']['application']['notification_body'].translate(str.maketrans('', '', "\r\n "))
|
current_system_title = self.data['settings']['application']['notification_body'].translate(TRANSLATE_WHITESPACE_TABLE)
|
||||||
for uuid, watch in self.data['watching'].items():
|
for uuid, watch in self.data['watching'].items():
|
||||||
try:
|
try:
|
||||||
watch_body = watch.get('notification_body', '')
|
watch_body = watch.get('notification_body', '')
|
||||||
if watch_body and watch_body.translate(str.maketrans('', '', "\r\n ")) == current_system_body:
|
if watch_body and watch_body.translate(TRANSLATE_WHITESPACE_TABLE) == current_system_body:
|
||||||
# Looks the same as the default one, so unset it
|
# Looks the same as the default one, so unset it
|
||||||
watch['notification_body'] = None
|
watch['notification_body'] = None
|
||||||
|
|
||||||
watch_title = watch.get('notification_title', '')
|
watch_title = watch.get('notification_title', '')
|
||||||
if watch_title and watch_title.translate(str.maketrans('', '', "\r\n ")) == current_system_title:
|
if watch_title and watch_title.translate(TRANSLATE_WHITESPACE_TABLE) == current_system_title:
|
||||||
# Looks the same as the default one, so unset it
|
# Looks the same as the default one, so unset it
|
||||||
watch['notification_title'] = None
|
watch['notification_title'] = None
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
|||||||
@@ -35,7 +35,9 @@
|
|||||||
|
|
||||||
<body class="">
|
<body class="">
|
||||||
<div class="header">
|
<div class="header">
|
||||||
<div class="home-menu pure-menu pure-menu-horizontal pure-menu-fixed" id="nav-menu">
|
<div class="pure-menu-fixed" style="width: 100%;">
|
||||||
|
<div class="home-menu pure-menu pure-menu-horizontal" id="nav-menu">
|
||||||
|
|
||||||
{% if has_password and not current_user.is_authenticated %}
|
{% if has_password and not current_user.is_authenticated %}
|
||||||
<a class="pure-menu-heading" href="https://changedetection.io" rel="noopener">
|
<a class="pure-menu-heading" href="https://changedetection.io" rel="noopener">
|
||||||
<strong>Change</strong>Detection.io</a>
|
<strong>Change</strong>Detection.io</a>
|
||||||
@@ -129,7 +131,12 @@
|
|||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
|
<div id="pure-menu-horizontal-spinner"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
{% if hosted_sticky %}
|
{% if hosted_sticky %}
|
||||||
<div class="sticky-tab" id="hosted-sticky">
|
<div class="sticky-tab" id="hosted-sticky">
|
||||||
<a href="https://changedetection.io/?ref={{guid}}">Let us host your instance!</a>
|
<a href="https://changedetection.io/?ref={{guid}}">Let us host your instance!</a>
|
||||||
|
|||||||
@@ -24,9 +24,8 @@
|
|||||||
const watch_visual_selector_data_url="{{url_for('static_content', group='visual_selector_data', filename=uuid)}}";
|
const watch_visual_selector_data_url="{{url_for('static_content', group='visual_selector_data', filename=uuid)}}";
|
||||||
const default_system_fetch_backend="{{ settings_application['fetch_backend'] }}";
|
const default_system_fetch_backend="{{ settings_application['fetch_backend'] }}";
|
||||||
</script>
|
</script>
|
||||||
|
<script src="{{url_for('static_content', group='js', filename='plugins.js')}}" defer></script>
|
||||||
<script src="{{url_for('static_content', group='js', filename='watch-settings.js')}}" defer></script>
|
<script src="{{url_for('static_content', group='js', filename='watch-settings.js')}}" defer></script>
|
||||||
<script src="{{url_for('static_content', group='js', filename='limit.js')}}" defer></script>
|
|
||||||
<script src="{{url_for('static_content', group='js', filename='notifications.js')}}" defer></script>
|
<script src="{{url_for('static_content', group='js', filename='notifications.js')}}" defer></script>
|
||||||
<script src="{{url_for('static_content', group='js', filename='visual-selector.js')}}" defer></script>
|
<script src="{{url_for('static_content', group='js', filename='visual-selector.js')}}" defer></script>
|
||||||
{% if playwright_enabled %}
|
{% if playwright_enabled %}
|
||||||
@@ -330,9 +329,9 @@ nav
|
|||||||
{{ render_checkbox_field(form.filter_text_added) }}
|
{{ render_checkbox_field(form.filter_text_added) }}
|
||||||
{{ render_checkbox_field(form.filter_text_replaced) }}
|
{{ render_checkbox_field(form.filter_text_replaced) }}
|
||||||
{{ render_checkbox_field(form.filter_text_removed) }}
|
{{ render_checkbox_field(form.filter_text_removed) }}
|
||||||
<span class="pure-form-message-inline">Note: Depending on the length and similarity of the text on each line, the algorithm may consider an <strong>addition</strong> instead of <strong>replacement</strong> for example.</span>
|
<span class="pure-form-message-inline">Note: Depending on the length and similarity of the text on each line, the algorithm may consider an <strong>addition</strong> instead of <strong>replacement</strong> for example.</span><br>
|
||||||
<span class="pure-form-message-inline">So it's always better to select <strong>Added</strong>+<strong>Replaced</strong> when you're interested in new content.</span><br>
|
<span class="pure-form-message-inline"> So it's always better to select <strong>Added</strong>+<strong>Replaced</strong> when you're interested in new content.</span><br>
|
||||||
<span class="pure-form-message-inline">When content is merely moved in a list, it will also trigger an <strong>addition</strong>, consider enabling <code><strong>Only trigger when unique lines appear</strong></code></span>
|
<span class="pure-form-message-inline"> When content is merely moved in a list, it will also trigger an <strong>addition</strong>, consider enabling <code><strong>Only trigger when unique lines appear</strong></code></span>
|
||||||
</fieldset>
|
</fieldset>
|
||||||
<fieldset class="pure-control-group">
|
<fieldset class="pure-control-group">
|
||||||
{{ render_checkbox_field(form.check_unique_lines) }}
|
{{ render_checkbox_field(form.check_unique_lines) }}
|
||||||
@@ -371,10 +370,10 @@ nav
|
|||||||
") }}
|
") }}
|
||||||
<span class="pure-form-message-inline">
|
<span class="pure-form-message-inline">
|
||||||
<ul>
|
<ul>
|
||||||
|
<li>Matching text will be <strong>ignored</strong> in the text snapshot (you can still see it but it wont trigger a change)</li>
|
||||||
<li>Each line processed separately, any line matching will be ignored (removed before creating the checksum)</li>
|
<li>Each line processed separately, any line matching will be ignored (removed before creating the checksum)</li>
|
||||||
<li>Regular Expression support, wrap the entire line in forward slash <code>/regex/</code></li>
|
<li>Regular Expression support, wrap the entire line in forward slash <code>/regex/</code></li>
|
||||||
<li>Changing this will affect the comparison checksum which may trigger an alert</li>
|
<li>Changing this will affect the comparison checksum which may trigger an alert</li>
|
||||||
<li>Use the preview/show current tab to see ignores</li>
|
|
||||||
</ul>
|
</ul>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
@@ -398,7 +397,9 @@ Unavailable") }}
|
|||||||
</fieldset>
|
</fieldset>
|
||||||
<fieldset>
|
<fieldset>
|
||||||
<div class="pure-control-group">
|
<div class="pure-control-group">
|
||||||
{{ render_field(form.extract_text, rows=5, placeholder="\d+ online") }}
|
{{ render_field(form.extract_text, rows=5, placeholder="/.+?\d+ comments.+?/
|
||||||
|
or
|
||||||
|
keyword") }}
|
||||||
<span class="pure-form-message-inline">
|
<span class="pure-form-message-inline">
|
||||||
<ul>
|
<ul>
|
||||||
<li>Extracts text in the final output (line by line) after other filters using regular expressions or string match;
|
<li>Extracts text in the final output (line by line) after other filters using regular expressions or string match;
|
||||||
@@ -422,14 +423,22 @@ Unavailable") }}
|
|||||||
<script>
|
<script>
|
||||||
const preview_text_edit_filters_url="{{url_for('watch_get_preview_rendered', uuid=uuid)}}";
|
const preview_text_edit_filters_url="{{url_for('watch_get_preview_rendered', uuid=uuid)}}";
|
||||||
</script>
|
</script>
|
||||||
<span><strong>Preview of the text that is used for changedetection after all filters run.</strong></span><br>
|
<br>
|
||||||
{#<div id="text-preview-controls"><span id="text-preview-refresh" class="pure-button button-xsmall">Refresh</span></div>#}
|
{#<div id="text-preview-controls"><span id="text-preview-refresh" class="pure-button button-xsmall">Refresh</span></div>#}
|
||||||
<p>
|
<div class="minitabs-wrapper">
|
||||||
<div id="text-preview-inner"></div>
|
<div class="minitabs-content">
|
||||||
</p>
|
<div id="text-preview-inner" class="monospace-preview">
|
||||||
|
<p>Loading...</p>
|
||||||
|
</div>
|
||||||
|
<div id="text-preview-before-inner" style="display: none;" class="monospace-preview">
|
||||||
|
<p>Loading...</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{# rendered sub Template #}
|
{# rendered sub Template #}
|
||||||
{% if extra_form_content %}
|
{% if extra_form_content %}
|
||||||
|
|||||||
@@ -3,11 +3,13 @@
|
|||||||
{% block content %}
|
{% block content %}
|
||||||
<script>
|
<script>
|
||||||
const screenshot_url = "{{url_for('static_content', group='screenshot', filename=uuid)}}";
|
const screenshot_url = "{{url_for('static_content', group='screenshot', filename=uuid)}}";
|
||||||
|
const triggered_line_numbers = {{ triggered_line_numbers|tojson }};
|
||||||
{% if last_error_screenshot %}
|
{% if last_error_screenshot %}
|
||||||
const error_screenshot_url = "{{url_for('static_content', group='screenshot', filename=uuid, error_screenshot=1) }}";
|
const error_screenshot_url = "{{url_for('static_content', group='screenshot', filename=uuid, error_screenshot=1) }}";
|
||||||
{% endif %}
|
{% endif %}
|
||||||
const highlight_submit_ignore_url = "{{url_for('highlight_submit_ignore_url', uuid=uuid)}}";
|
const highlight_submit_ignore_url = "{{url_for('highlight_submit_ignore_url', uuid=uuid)}}";
|
||||||
</script>
|
</script>
|
||||||
|
<script src="{{url_for('static_content', group='js', filename='plugins.js')}}"></script>
|
||||||
<script src="{{ url_for('static_content', group='js', filename='diff-overview.js') }}" defer></script>
|
<script src="{{ url_for('static_content', group='js', filename='diff-overview.js') }}" defer></script>
|
||||||
<script src="{{ url_for('static_content', group='js', filename='preview.js') }}" defer></script>
|
<script src="{{ url_for('static_content', group='js', filename='preview.js') }}" defer></script>
|
||||||
<script src="{{ url_for('static_content', group='js', filename='tabs.js') }}" defer></script>
|
<script src="{{ url_for('static_content', group='js', filename='tabs.js') }}" defer></script>
|
||||||
@@ -67,16 +69,15 @@
|
|||||||
|
|
||||||
<div class="tab-pane-inner" id="text">
|
<div class="tab-pane-inner" id="text">
|
||||||
<div class="snapshot-age">{{ current_version|format_timestamp_timeago }}</div>
|
<div class="snapshot-age">{{ current_version|format_timestamp_timeago }}</div>
|
||||||
<span class="ignored">Grey lines are ignored</span> <span class="triggered">Blue lines are triggers</span>
|
|
||||||
<span class="tip"><strong>Pro-tip</strong>: Highlight text to add to ignore filters</span>
|
<span class="tip"><strong>Pro-tip</strong>: Highlight text to add to ignore filters</span>
|
||||||
|
|
||||||
<table>
|
<table>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<td id="diff-col" class="highlightable-filter">
|
<td id="diff-col" class="highlightable-filter">
|
||||||
{% for row in content %}
|
<pre style="border-left: 2px solid #ddd;">
|
||||||
<div class="{{ row.classes }}">{{ row.line }}</div>
|
{{ content }}
|
||||||
{% endfor %}
|
</pre>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
|||||||
@@ -129,6 +129,13 @@
|
|||||||
Note: Simply changing the User-Agent often does not defeat anti-robot technologies, it's important to consider <a href="https://changedetection.io/tutorial/what-are-main-types-anti-robot-mechanisms">all of the ways that the browser is detected</a>.
|
Note: Simply changing the User-Agent often does not defeat anti-robot technologies, it's important to consider <a href="https://changedetection.io/tutorial/what-are-main-types-anti-robot-mechanisms">all of the ways that the browser is detected</a>.
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="pure-control-group">
|
||||||
|
{{ render_field(form.application.form.keep_history_n) }}
|
||||||
|
<span class="pure-form-message-inline">Blank - keep all</span>
|
||||||
|
{{ render_field(form.application.form.keep_history_seconds) }}
|
||||||
|
<span class="pure-form-message-inline">Blank - keep all</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="pure-control-group">
|
<div class="pure-control-group">
|
||||||
<br>
|
<br>
|
||||||
Tip: <a href="https://github.com/dgtlmoon/changedetection.io/wiki/Proxy-configuration#brightdata-proxy-support">Connect using Bright Data and Oxylabs Proxies, find out more here.</a>
|
Tip: <a href="https://github.com/dgtlmoon/changedetection.io/wiki/Proxy-configuration#brightdata-proxy-support">Connect using Bright Data and Oxylabs Proxies, find out more here.</a>
|
||||||
@@ -172,11 +179,11 @@ nav
|
|||||||
<span class="pure-form-message-inline">Note: This is applied globally in addition to the per-watch rules.</span><br>
|
<span class="pure-form-message-inline">Note: This is applied globally in addition to the per-watch rules.</span><br>
|
||||||
<span class="pure-form-message-inline">
|
<span class="pure-form-message-inline">
|
||||||
<ul>
|
<ul>
|
||||||
|
<li>Matching text will be <strong>ignored</strong> in the text snapshot (you can still see it but it wont trigger a change)</li>
|
||||||
<li>Note: This is applied globally in addition to the per-watch rules.</li>
|
<li>Note: This is applied globally in addition to the per-watch rules.</li>
|
||||||
<li>Each line processed separately, any line matching will be ignored (removed before creating the checksum)</li>
|
<li>Each line processed separately, any line matching will be ignored (removed before creating the checksum)</li>
|
||||||
<li>Regular Expression support, wrap the entire line in forward slash <code>/regex/</code></li>
|
<li>Regular Expression support, wrap the entire line in forward slash <code>/regex/</code></li>
|
||||||
<li>Changing this will affect the comparison checksum which may trigger an alert</li>
|
<li>Changing this will affect the comparison checksum which may trigger an alert</li>
|
||||||
<li>Use the preview/show current tab to see ignores</li>
|
|
||||||
</ul>
|
</ul>
|
||||||
</span>
|
</span>
|
||||||
</fieldset>
|
</fieldset>
|
||||||
|
|||||||
@@ -78,8 +78,8 @@
|
|||||||
{% if any_has_restock_price_processor %}
|
{% if any_has_restock_price_processor %}
|
||||||
<th>Restock & Price</th>
|
<th>Restock & Price</th>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<th><a class="{{ 'active '+link_order if sort_attribute == 'last_checked' else 'inactive' }}" href="{{url_for('index', sort='last_checked', order=link_order, tag=active_tag_uuid)}}">Last Checked <span class='arrow {{link_order}}'></span></a></th>
|
<th><a class="{{ 'active '+link_order if sort_attribute == 'last_checked' else 'inactive' }}" href="{{url_for('index', sort='last_checked', order=link_order, tag=active_tag_uuid)}}"><span class="hide-on-mobile">Last</span> Checked <span class='arrow {{link_order}}'></span></a></th>
|
||||||
<th><a class="{{ 'active '+link_order if sort_attribute == 'last_changed' else 'inactive' }}" href="{{url_for('index', sort='last_changed', order=link_order, tag=active_tag_uuid)}}">Last Changed <span class='arrow {{link_order}}'></span></a></th>
|
<th><a class="{{ 'active '+link_order if sort_attribute == 'last_changed' else 'inactive' }}" href="{{url_for('index', sort='last_changed', order=link_order, tag=active_tag_uuid)}}"><span class="hide-on-mobile">Last</span> Changed <span class='arrow {{link_order}}'></span></a></th>
|
||||||
<th class="empty-cell"></th>
|
<th class="empty-cell"></th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
@@ -191,9 +191,9 @@
|
|||||||
{% if watch.history_n >= 2 %}
|
{% if watch.history_n >= 2 %}
|
||||||
|
|
||||||
{% if is_unviewed %}
|
{% if is_unviewed %}
|
||||||
<a href="{{ url_for('diff_history_page', uuid=watch.uuid, from_version=watch.get_next_snapshot_key_to_last_viewed) }}" target="{{watch.uuid}}" class="pure-button pure-button-primary diff-link">Diff</a>
|
<a href="{{ url_for('diff_history_page', uuid=watch.uuid, from_version=watch.get_next_snapshot_key_to_last_viewed) }}" target="{{watch.uuid}}" class="pure-button pure-button-primary diff-link">History</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
<a href="{{ url_for('diff_history_page', uuid=watch.uuid)}}" target="{{watch.uuid}}" class="pure-button pure-button-primary diff-link">Diff</a>
|
<a href="{{ url_for('diff_history_page', uuid=watch.uuid)}}" target="{{watch.uuid}}" class="pure-button pure-button-primary diff-link">History</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% else %}
|
{% else %}
|
||||||
|
|||||||
6
changedetectionio/tests/itemprop_test_examples/README.md
Normal file
6
changedetectionio/tests/itemprop_test_examples/README.md
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
# A list of real world examples!
|
||||||
|
|
||||||
|
Always the price should be 666.66 for our tests
|
||||||
|
|
||||||
|
see test_restock_itemprop.py::test_special_prop_examples
|
||||||
|
|
||||||
25
changedetectionio/tests/itemprop_test_examples/a.txt
Normal file
25
changedetectionio/tests/itemprop_test_examples/a.txt
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
<div class="PriceSection PriceSection_PriceSection__Vx1_Q PriceSection_variantHuge__P9qxg PdpPriceSection"
|
||||||
|
data-testid="price-section"
|
||||||
|
data-optly-product-tile-price-section="true"><span
|
||||||
|
class="PriceRange ProductPrice variant-huge" itemprop="offers"
|
||||||
|
itemscope="" itemtype="http://schema.org/Offer"><div
|
||||||
|
class="VisuallyHidden_VisuallyHidden__VBD83">$155.55</div><span
|
||||||
|
aria-hidden="true" class="Price variant-huge" data-testid="price"
|
||||||
|
itemprop="price"><sup class="sup" data-testid="price-symbol"
|
||||||
|
itemprop="priceCurrency" content="AUD">$</sup><span
|
||||||
|
class="dollars" data-testid="price-value" itemprop="price"
|
||||||
|
content="155.55">155.55</span><span class="extras"><span class="sup"
|
||||||
|
data-testid="price-sup"></span></span></span></span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="application/ld+json">{
|
||||||
|
"@type": "Product",
|
||||||
|
"@context": "https://schema.org",
|
||||||
|
"name": "test",
|
||||||
|
"description": "test",
|
||||||
|
"offers": {
|
||||||
|
"@type": "Offer",
|
||||||
|
"priceCurrency": "AUD",
|
||||||
|
"price": 155.55
|
||||||
|
},
|
||||||
|
}</script>
|
||||||
@@ -16,4 +16,4 @@ def test_check_basic_change_detection_functionality(client, live_server, measure
|
|||||||
)
|
)
|
||||||
|
|
||||||
assert b"1 Imported" in res.data
|
assert b"1 Imported" in res.data
|
||||||
time.sleep(3)
|
wait_for_all_checks(client)
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ def test_select_custom(client, live_server, measure_memory_usage):
|
|||||||
follow_redirects=True
|
follow_redirects=True
|
||||||
)
|
)
|
||||||
# We should see something via proxy
|
# We should see something via proxy
|
||||||
assert b'<div class=""> - 0.' in res.data
|
assert b' - 0.' in res.data
|
||||||
|
|
||||||
#
|
#
|
||||||
# Now we should see the request in the container logs for "squid-squid-custom" because it will be the only default
|
# Now we should see the request in the container logs for "squid-squid-custom" because it will be the only default
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
import json
|
||||||
import os
|
import os
|
||||||
from flask import url_for
|
from flask import url_for
|
||||||
from changedetectionio.tests.util import live_server_setup, wait_for_all_checks
|
from changedetectionio.tests.util import live_server_setup, wait_for_all_checks, extract_UUID_from_client
|
||||||
|
|
||||||
|
|
||||||
def set_response():
|
def set_response():
|
||||||
@@ -18,7 +19,6 @@ def set_response():
|
|||||||
f.write(data)
|
f.write(data)
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
|
|
||||||
|
|
||||||
def test_socks5(client, live_server, measure_memory_usage):
|
def test_socks5(client, live_server, measure_memory_usage):
|
||||||
live_server_setup(live_server)
|
live_server_setup(live_server)
|
||||||
set_response()
|
set_response()
|
||||||
@@ -79,3 +79,24 @@ def test_socks5(client, live_server, measure_memory_usage):
|
|||||||
|
|
||||||
# Should see the proper string
|
# Should see the proper string
|
||||||
assert "Awesome, you made it".encode('utf-8') in res.data
|
assert "Awesome, you made it".encode('utf-8') in res.data
|
||||||
|
|
||||||
|
# PROXY CHECKER WIDGET CHECK - this needs more checking
|
||||||
|
uuid = extract_UUID_from_client(client)
|
||||||
|
|
||||||
|
res = client.get(
|
||||||
|
url_for("check_proxies.start_check", uuid=uuid),
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
# It's probably already finished super fast :(
|
||||||
|
#assert b"RUNNING" in res.data
|
||||||
|
|
||||||
|
wait_for_all_checks(client)
|
||||||
|
res = client.get(
|
||||||
|
url_for("check_proxies.get_recheck_status", uuid=uuid),
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
assert b"OK" in res.data
|
||||||
|
|
||||||
|
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
||||||
|
assert b'Deleted' in res.data
|
||||||
|
|
||||||
|
|||||||
@@ -39,9 +39,8 @@ def test_setup(client, live_server, measure_memory_usage):
|
|||||||
live_server_setup(live_server)
|
live_server_setup(live_server)
|
||||||
|
|
||||||
def test_check_removed_line_contains_trigger(client, live_server, measure_memory_usage):
|
def test_check_removed_line_contains_trigger(client, live_server, measure_memory_usage):
|
||||||
|
#live_server_setup(live_server)
|
||||||
# Give the endpoint time to spin up
|
# Give the endpoint time to spin up
|
||||||
time.sleep(1)
|
|
||||||
set_original()
|
set_original()
|
||||||
# Add our URL to the import page
|
# Add our URL to the import page
|
||||||
test_url = url_for('test_endpoint', _external=True)
|
test_url = url_for('test_endpoint', _external=True)
|
||||||
@@ -78,6 +77,8 @@ def test_check_removed_line_contains_trigger(client, live_server, measure_memory
|
|||||||
|
|
||||||
# The trigger line is REMOVED, this should trigger
|
# The trigger line is REMOVED, this should trigger
|
||||||
set_original(excluding='The golden line')
|
set_original(excluding='The golden line')
|
||||||
|
|
||||||
|
# Check in the processor here what's going on, its triggering empty-reply and no change.
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
wait_for_all_checks(client)
|
wait_for_all_checks(client)
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
@@ -153,6 +154,7 @@ def test_check_add_line_contains_trigger(client, live_server, measure_memory_usa
|
|||||||
# A line thats not the trigger should not trigger anything
|
# A line thats not the trigger should not trigger anything
|
||||||
res = client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
res = client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
assert b'1 watches queued for rechecking.' in res.data
|
assert b'1 watches queued for rechecking.' in res.data
|
||||||
|
|
||||||
wait_for_all_checks(client)
|
wait_for_all_checks(client)
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
assert b'unviewed' not in res.data
|
assert b'unviewed' not in res.data
|
||||||
@@ -172,6 +174,5 @@ def test_check_add_line_contains_trigger(client, live_server, measure_memory_usa
|
|||||||
assert b'-Oh yes please-' in response
|
assert b'-Oh yes please-' in response
|
||||||
assert '网站监测 内容更新了'.encode('utf-8') in response
|
assert '网站监测 内容更新了'.encode('utf-8') in response
|
||||||
|
|
||||||
|
|
||||||
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
||||||
assert b'Deleted' in res.data
|
assert b'Deleted' in res.data
|
||||||
|
|||||||
@@ -65,11 +65,8 @@ def test_check_block_changedetection_text_NOT_present(client, live_server, measu
|
|||||||
live_server_setup(live_server)
|
live_server_setup(live_server)
|
||||||
# Use a mix of case in ZzZ to prove it works case-insensitive.
|
# Use a mix of case in ZzZ to prove it works case-insensitive.
|
||||||
ignore_text = "out of stoCk\r\nfoobar"
|
ignore_text = "out of stoCk\r\nfoobar"
|
||||||
|
|
||||||
set_original_ignore_response()
|
set_original_ignore_response()
|
||||||
|
|
||||||
# Give the endpoint time to spin up
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
# Add our URL to the import page
|
# Add our URL to the import page
|
||||||
test_url = url_for('test_endpoint', _external=True)
|
test_url = url_for('test_endpoint', _external=True)
|
||||||
@@ -127,13 +124,24 @@ def test_check_block_changedetection_text_NOT_present(client, live_server, measu
|
|||||||
assert b'unviewed' not in res.data
|
assert b'unviewed' not in res.data
|
||||||
assert b'/test-endpoint' in res.data
|
assert b'/test-endpoint' in res.data
|
||||||
|
|
||||||
|
# 2548
|
||||||
|
# Going back to the ORIGINAL should NOT trigger a change
|
||||||
|
set_original_ignore_response()
|
||||||
|
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
|
wait_for_all_checks(client)
|
||||||
|
res = client.get(url_for("index"))
|
||||||
|
assert b'unviewed' not in res.data
|
||||||
|
|
||||||
# Now we set a change where the text is gone, it should now trigger
|
|
||||||
|
# Now we set a change where the text is gone AND its different content, it should now trigger
|
||||||
set_modified_response_minus_block_text()
|
set_modified_response_minus_block_text()
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
wait_for_all_checks(client)
|
wait_for_all_checks(client)
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
assert b'unviewed' in res.data
|
assert b'unviewed' in res.data
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
||||||
assert b'Deleted' in res.data
|
assert b'Deleted' in res.data
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import time
|
|||||||
from flask import url_for
|
from flask import url_for
|
||||||
|
|
||||||
from ..html_tools import *
|
from ..html_tools import *
|
||||||
from .util import live_server_setup
|
from .util import live_server_setup, wait_for_all_checks
|
||||||
|
|
||||||
|
|
||||||
def test_setup(live_server):
|
def test_setup(live_server):
|
||||||
@@ -119,12 +119,10 @@ across multiple lines
|
|||||||
|
|
||||||
|
|
||||||
def test_element_removal_full(client, live_server, measure_memory_usage):
|
def test_element_removal_full(client, live_server, measure_memory_usage):
|
||||||
sleep_time_for_fetch_thread = 3
|
#live_server_setup(live_server)
|
||||||
|
|
||||||
set_original_response()
|
set_original_response()
|
||||||
|
|
||||||
# Give the endpoint time to spin up
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
# Add our URL to the import page
|
# Add our URL to the import page
|
||||||
test_url = url_for("test_endpoint", _external=True)
|
test_url = url_for("test_endpoint", _external=True)
|
||||||
@@ -132,7 +130,8 @@ def test_element_removal_full(client, live_server, measure_memory_usage):
|
|||||||
url_for("import_page"), data={"urls": test_url}, follow_redirects=True
|
url_for("import_page"), data={"urls": test_url}, follow_redirects=True
|
||||||
)
|
)
|
||||||
assert b"1 Imported" in res.data
|
assert b"1 Imported" in res.data
|
||||||
time.sleep(1)
|
wait_for_all_checks(client)
|
||||||
|
|
||||||
# Goto the edit page, add the filter data
|
# Goto the edit page, add the filter data
|
||||||
# Not sure why \r needs to be added - absent of the #changetext this is not necessary
|
# Not sure why \r needs to be added - absent of the #changetext this is not necessary
|
||||||
subtractive_selectors_data = "header\r\nfooter\r\nnav\r\n#changetext"
|
subtractive_selectors_data = "header\r\nfooter\r\nnav\r\n#changetext"
|
||||||
@@ -148,6 +147,7 @@ def test_element_removal_full(client, live_server, measure_memory_usage):
|
|||||||
follow_redirects=True,
|
follow_redirects=True,
|
||||||
)
|
)
|
||||||
assert b"Updated watch." in res.data
|
assert b"Updated watch." in res.data
|
||||||
|
wait_for_all_checks(client)
|
||||||
|
|
||||||
# Check it saved
|
# Check it saved
|
||||||
res = client.get(
|
res = client.get(
|
||||||
@@ -156,10 +156,10 @@ def test_element_removal_full(client, live_server, measure_memory_usage):
|
|||||||
assert bytes(subtractive_selectors_data.encode("utf-8")) in res.data
|
assert bytes(subtractive_selectors_data.encode("utf-8")) in res.data
|
||||||
|
|
||||||
# Trigger a check
|
# Trigger a check
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
res = client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
|
assert b'1 watches queued for rechecking.' in res.data
|
||||||
|
|
||||||
# Give the thread time to pick it up
|
wait_for_all_checks(client)
|
||||||
time.sleep(sleep_time_for_fetch_thread)
|
|
||||||
|
|
||||||
# so that we set the state to 'unviewed' after all the edits
|
# so that we set the state to 'unviewed' after all the edits
|
||||||
client.get(url_for("diff_history_page", uuid="first"))
|
client.get(url_for("diff_history_page", uuid="first"))
|
||||||
@@ -168,10 +168,11 @@ def test_element_removal_full(client, live_server, measure_memory_usage):
|
|||||||
set_modified_response()
|
set_modified_response()
|
||||||
|
|
||||||
# Trigger a check
|
# Trigger a check
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
res = client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
|
assert b'1 watches queued for rechecking.' in res.data
|
||||||
|
|
||||||
# Give the thread time to pick it up
|
# Give the thread time to pick it up
|
||||||
time.sleep(sleep_time_for_fetch_thread)
|
wait_for_all_checks(client)
|
||||||
|
|
||||||
# There should not be an unviewed change, as changes should be removed
|
# There should not be an unviewed change, as changes should be removed
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
import time
|
import time
|
||||||
from flask import url_for
|
from flask import url_for
|
||||||
from .util import live_server_setup, wait_for_all_checks
|
from .util import live_server_setup, wait_for_all_checks, extract_UUID_from_client
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
|
|
||||||
@@ -38,6 +38,11 @@ def test_check_encoding_detection(client, live_server, measure_memory_usage):
|
|||||||
# Give the thread time to pick it up
|
# Give the thread time to pick it up
|
||||||
wait_for_all_checks(client)
|
wait_for_all_checks(client)
|
||||||
|
|
||||||
|
|
||||||
|
# Content type recording worked
|
||||||
|
uuid = extract_UUID_from_client(client)
|
||||||
|
assert live_server.app.config['DATASTORE'].data['watching'][uuid]['content-type'] == "text/html"
|
||||||
|
|
||||||
res = client.get(
|
res = client.get(
|
||||||
url_for("preview_page", uuid="first"),
|
url_for("preview_page", uuid="first"),
|
||||||
follow_redirects=True
|
follow_redirects=True
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ def test_setup(client, live_server, measure_memory_usage):
|
|||||||
live_server_setup(live_server)
|
live_server_setup(live_server)
|
||||||
|
|
||||||
def test_check_filter_multiline(client, live_server, measure_memory_usage):
|
def test_check_filter_multiline(client, live_server, measure_memory_usage):
|
||||||
#live_server_setup(live_server)
|
# live_server_setup(live_server)
|
||||||
set_multiline_response()
|
set_multiline_response()
|
||||||
|
|
||||||
# Add our URL to the import page
|
# Add our URL to the import page
|
||||||
@@ -115,9 +115,9 @@ def test_check_filter_multiline(client, live_server, measure_memory_usage):
|
|||||||
# Plaintext that doesnt look like a regex should match also
|
# Plaintext that doesnt look like a regex should match also
|
||||||
assert b'and this should be' in res.data
|
assert b'and this should be' in res.data
|
||||||
|
|
||||||
assert b'<div class="">Something' in res.data
|
assert b'Something' in res.data
|
||||||
assert b'<div class="">across 6 billion multiple' in res.data
|
assert b'across 6 billion multiple' in res.data
|
||||||
assert b'<div class="">lines' in res.data
|
assert b'lines' in res.data
|
||||||
|
|
||||||
# but the last one, which also says 'lines' shouldnt be here (non-greedy match checking)
|
# but the last one, which also says 'lines' shouldnt be here (non-greedy match checking)
|
||||||
assert b'aaand something lines' not in res.data
|
assert b'aaand something lines' not in res.data
|
||||||
@@ -183,20 +183,19 @@ def test_check_filter_and_regex_extract(client, live_server, measure_memory_usag
|
|||||||
follow_redirects=True
|
follow_redirects=True
|
||||||
)
|
)
|
||||||
|
|
||||||
# Class will be blank for now because the frontend didnt apply the diff
|
assert b'1000 online' in res.data
|
||||||
assert b'<div class="">1000 online' in res.data
|
|
||||||
|
|
||||||
# All regex matching should be here
|
# All regex matching should be here
|
||||||
assert b'<div class="">2000 online' in res.data
|
assert b'2000 online' in res.data
|
||||||
|
|
||||||
# Both regexs should be here
|
# Both regexs should be here
|
||||||
assert b'<div class="">80 guests' in res.data
|
assert b'80 guests' in res.data
|
||||||
|
|
||||||
# Regex with flag handling should be here
|
# Regex with flag handling should be here
|
||||||
assert b'<div class="">SomeCase insensitive 3456' in res.data
|
assert b'SomeCase insensitive 3456' in res.data
|
||||||
|
|
||||||
# Singular group from /somecase insensitive (345\d)/i
|
# Singular group from /somecase insensitive (345\d)/i
|
||||||
assert b'<div class="">3456' in res.data
|
assert b'3456' in res.data
|
||||||
|
|
||||||
# Regex with multiline flag handling should be here
|
# Regex with multiline flag handling should be here
|
||||||
|
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ def set_original_ignore_response():
|
|||||||
f.write(test_return_data)
|
f.write(test_return_data)
|
||||||
|
|
||||||
|
|
||||||
def test_highlight_ignore(client, live_server, measure_memory_usage):
|
def test_ignore(client, live_server, measure_memory_usage):
|
||||||
live_server_setup(live_server)
|
live_server_setup(live_server)
|
||||||
set_original_ignore_response()
|
set_original_ignore_response()
|
||||||
test_url = url_for('test_endpoint', _external=True)
|
test_url = url_for('test_endpoint', _external=True)
|
||||||
@@ -51,9 +51,9 @@ def test_highlight_ignore(client, live_server, measure_memory_usage):
|
|||||||
# Should return a link
|
# Should return a link
|
||||||
assert b'href' in res.data
|
assert b'href' in res.data
|
||||||
|
|
||||||
# And it should register in the preview page
|
# It should not be in the preview anymore
|
||||||
res = client.get(url_for("preview_page", uuid=uuid))
|
res = client.get(url_for("preview_page", uuid=uuid))
|
||||||
assert b'<div class="ignored">oh yeah 456' in res.data
|
assert b'<div class="ignored">oh yeah 456' not in res.data
|
||||||
|
|
||||||
# Should be in base.html
|
# Should be in base.html
|
||||||
assert b'csrftoken' in res.data
|
assert b'csrftoken' in res.data
|
||||||
@@ -33,13 +33,17 @@ def test_strip_regex_text_func():
|
|||||||
|
|
||||||
stripped_content = html_tools.strip_ignore_text(test_content, ignore_lines)
|
stripped_content = html_tools.strip_ignore_text(test_content, ignore_lines)
|
||||||
|
|
||||||
assert b"but 1 lines" in stripped_content
|
assert "but 1 lines" in stripped_content
|
||||||
assert b"igNORe-cAse text" not in stripped_content
|
assert "igNORe-cAse text" not in stripped_content
|
||||||
assert b"but 1234 lines" not in stripped_content
|
assert "but 1234 lines" not in stripped_content
|
||||||
assert b"really" not in stripped_content
|
assert "really" not in stripped_content
|
||||||
assert b"not this" not in stripped_content
|
assert "not this" not in stripped_content
|
||||||
|
|
||||||
# Check line number reporting
|
# Check line number reporting
|
||||||
stripped_content = html_tools.strip_ignore_text(test_content, ignore_lines, mode="line numbers")
|
stripped_content = html_tools.strip_ignore_text(test_content, ignore_lines, mode="line numbers")
|
||||||
assert stripped_content == [2, 5, 6, 7, 8, 10]
|
assert stripped_content == [2, 5, 6, 7, 8, 10]
|
||||||
|
|
||||||
|
# Check that linefeeds are preserved when there are is no matching ignores
|
||||||
|
content = "some text\n\nand other text\n"
|
||||||
|
stripped_content = html_tools.strip_ignore_text(content, ignore_lines)
|
||||||
|
assert content == stripped_content
|
||||||
|
|||||||
@@ -22,10 +22,15 @@ def test_strip_text_func():
|
|||||||
ignore_lines = ["sometimes"]
|
ignore_lines = ["sometimes"]
|
||||||
|
|
||||||
stripped_content = html_tools.strip_ignore_text(test_content, ignore_lines)
|
stripped_content = html_tools.strip_ignore_text(test_content, ignore_lines)
|
||||||
|
assert "sometimes" not in stripped_content
|
||||||
|
assert "Some content" in stripped_content
|
||||||
|
|
||||||
assert b"sometimes" not in stripped_content
|
# Check that line feeds dont get chewed up when something is found
|
||||||
assert b"Some content" in stripped_content
|
test_content = "Some initial text\n\nWhich is across multiple lines\n\nZZZZz\n\n\nSo let's see what happens."
|
||||||
|
ignore = ['something irrelevent but just to check', 'XXXXX', 'YYYYY', 'ZZZZZ']
|
||||||
|
|
||||||
|
stripped_content = html_tools.strip_ignore_text(test_content, ignore)
|
||||||
|
assert stripped_content == "Some initial text\n\nWhich is across multiple lines\n\n\n\nSo let's see what happens."
|
||||||
|
|
||||||
def set_original_ignore_response():
|
def set_original_ignore_response():
|
||||||
test_return_data = """<html>
|
test_return_data = """<html>
|
||||||
@@ -79,14 +84,14 @@ def set_modified_ignore_response():
|
|||||||
f.write(test_return_data)
|
f.write(test_return_data)
|
||||||
|
|
||||||
|
|
||||||
|
# Ignore text now just removes it entirely, is a LOT more simpler code this way
|
||||||
|
|
||||||
def test_check_ignore_text_functionality(client, live_server, measure_memory_usage):
|
def test_check_ignore_text_functionality(client, live_server, measure_memory_usage):
|
||||||
|
|
||||||
# Use a mix of case in ZzZ to prove it works case-insensitive.
|
# Use a mix of case in ZzZ to prove it works case-insensitive.
|
||||||
ignore_text = "XXXXX\r\nYYYYY\r\nzZzZZ\r\nnew ignore stuff"
|
ignore_text = "XXXXX\r\nYYYYY\r\nzZzZZ\r\nnew ignore stuff"
|
||||||
set_original_ignore_response()
|
set_original_ignore_response()
|
||||||
|
|
||||||
# Give the endpoint time to spin up
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
# Add our URL to the import page
|
# Add our URL to the import page
|
||||||
test_url = url_for('test_endpoint', _external=True)
|
test_url = url_for('test_endpoint', _external=True)
|
||||||
@@ -141,8 +146,6 @@ def test_check_ignore_text_functionality(client, live_server, measure_memory_usa
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# Just to be sure.. set a regular modified change..
|
# Just to be sure.. set a regular modified change..
|
||||||
set_modified_original_ignore_response()
|
set_modified_original_ignore_response()
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
@@ -151,21 +154,19 @@ def test_check_ignore_text_functionality(client, live_server, measure_memory_usa
|
|||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
assert b'unviewed' in res.data
|
assert b'unviewed' in res.data
|
||||||
|
|
||||||
# Check the preview/highlighter, we should be able to see what we ignored, but it should be highlighted
|
|
||||||
# We only introduce the "modified" content that includes what we ignore so we can prove the newest version also displays
|
|
||||||
# at /preview
|
|
||||||
res = client.get(url_for("preview_page", uuid="first"))
|
res = client.get(url_for("preview_page", uuid="first"))
|
||||||
# We should be able to see what we ignored
|
|
||||||
assert b'<div class="ignored">new ignore stuff' in res.data
|
# SHOULD BE be in the preview, it was added in set_modified_original_ignore_response()
|
||||||
|
# and we have "new ignore stuff" in ignore_text
|
||||||
|
# it is only ignored, it is not removed (it will be highlighted too)
|
||||||
|
assert b'new ignore stuff' in res.data
|
||||||
|
|
||||||
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
||||||
assert b'Deleted' in res.data
|
assert b'Deleted' in res.data
|
||||||
|
|
||||||
|
# When adding some ignore text, it should not trigger a change, even if something else on that line changes
|
||||||
def test_check_global_ignore_text_functionality(client, live_server, measure_memory_usage):
|
def test_check_global_ignore_text_functionality(client, live_server, measure_memory_usage):
|
||||||
|
#live_server_setup(live_server)
|
||||||
# Give the endpoint time to spin up
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
ignore_text = "XXXXX\r\nYYYYY\r\nZZZZZ"
|
ignore_text = "XXXXX\r\nYYYYY\r\nZZZZZ"
|
||||||
set_original_ignore_response()
|
set_original_ignore_response()
|
||||||
|
|
||||||
@@ -174,6 +175,7 @@ def test_check_global_ignore_text_functionality(client, live_server, measure_mem
|
|||||||
url_for("settings_page"),
|
url_for("settings_page"),
|
||||||
data={
|
data={
|
||||||
"requests-time_between_check-minutes": 180,
|
"requests-time_between_check-minutes": 180,
|
||||||
|
"application-ignore_whitespace": "y",
|
||||||
"application-global_ignore_text": ignore_text,
|
"application-global_ignore_text": ignore_text,
|
||||||
'application-fetch_backend': "html_requests"
|
'application-fetch_backend': "html_requests"
|
||||||
},
|
},
|
||||||
@@ -194,9 +196,7 @@ def test_check_global_ignore_text_functionality(client, live_server, measure_mem
|
|||||||
# Give the thread time to pick it up
|
# Give the thread time to pick it up
|
||||||
wait_for_all_checks(client)
|
wait_for_all_checks(client)
|
||||||
|
|
||||||
|
#Adding some ignore text should not trigger a change
|
||||||
# Goto the edit page of the item, add our ignore text
|
|
||||||
# Add our URL to the import page
|
|
||||||
res = client.post(
|
res = client.post(
|
||||||
url_for("edit_page", uuid="first"),
|
url_for("edit_page", uuid="first"),
|
||||||
data={"ignore_text": "something irrelevent but just to check", "url": test_url, 'fetch_backend': "html_requests"},
|
data={"ignore_text": "something irrelevent but just to check", "url": test_url, 'fetch_backend': "html_requests"},
|
||||||
@@ -212,20 +212,15 @@ def test_check_global_ignore_text_functionality(client, live_server, measure_mem
|
|||||||
|
|
||||||
# Trigger a check
|
# Trigger a check
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
|
|
||||||
# Give the thread time to pick it up
|
|
||||||
wait_for_all_checks(client)
|
wait_for_all_checks(client)
|
||||||
|
# It should report nothing found (no new 'unviewed' class), adding random ignore text should not cause a change
|
||||||
# so that we are sure everything is viewed and in a known 'nothing changed' state
|
|
||||||
res = client.get(url_for("diff_history_page", uuid="first"))
|
|
||||||
|
|
||||||
# It should report nothing found (no new 'unviewed' class)
|
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
assert b'unviewed' not in res.data
|
assert b'unviewed' not in res.data
|
||||||
assert b'/test-endpoint' in res.data
|
assert b'/test-endpoint' in res.data
|
||||||
|
#####
|
||||||
|
|
||||||
|
# Make a change which includes the ignore text, it should be ignored and no 'change' triggered
|
||||||
# Make a change which includes the ignore text
|
# It adds text with "ZZZZzzzz" and "ZZZZ" is in the ignore list
|
||||||
set_modified_ignore_response()
|
set_modified_ignore_response()
|
||||||
|
|
||||||
# Trigger a check
|
# Trigger a check
|
||||||
@@ -235,6 +230,7 @@ def test_check_global_ignore_text_functionality(client, live_server, measure_mem
|
|||||||
|
|
||||||
# It should report nothing found (no new 'unviewed' class)
|
# It should report nothing found (no new 'unviewed' class)
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
|
|
||||||
assert b'unviewed' not in res.data
|
assert b'unviewed' not in res.data
|
||||||
assert b'/test-endpoint' in res.data
|
assert b'/test-endpoint' in res.data
|
||||||
|
|
||||||
|
|||||||
@@ -499,7 +499,7 @@ def test_correct_header_detect(client, live_server, measure_memory_usage):
|
|||||||
)
|
)
|
||||||
|
|
||||||
assert b'"hello": 123,' in res.data
|
assert b'"hello": 123,' in res.data
|
||||||
assert b'"world": 123</div>' in res.data
|
assert b'"world": 123' in res.data
|
||||||
|
|
||||||
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
||||||
assert b'Deleted' in res.data
|
assert b'Deleted' in res.data
|
||||||
|
|||||||
78
changedetectionio/tests/test_live_preview.py
Normal file
78
changedetectionio/tests/test_live_preview.py
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
from flask import url_for
|
||||||
|
from changedetectionio.tests.util import live_server_setup, wait_for_all_checks, extract_UUID_from_client
|
||||||
|
|
||||||
|
|
||||||
|
def set_response():
|
||||||
|
|
||||||
|
data = f"""<html>
|
||||||
|
<body>Awesome, you made it<br>
|
||||||
|
yeah the socks request worked<br>
|
||||||
|
something to ignore<br>
|
||||||
|
something to trigger<br>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
"""
|
||||||
|
|
||||||
|
with open("test-datastore/endpoint-content.txt", "w") as f:
|
||||||
|
f.write(data)
|
||||||
|
|
||||||
|
def test_content_filter_live_preview(client, live_server, measure_memory_usage):
|
||||||
|
live_server_setup(live_server)
|
||||||
|
set_response()
|
||||||
|
|
||||||
|
test_url = url_for('test_endpoint', _external=True)
|
||||||
|
|
||||||
|
res = client.post(
|
||||||
|
url_for("form_quick_watch_add"),
|
||||||
|
data={"url": test_url, "tags": ''},
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
uuid = extract_UUID_from_client(client)
|
||||||
|
res = client.post(
|
||||||
|
url_for("edit_page", uuid=uuid),
|
||||||
|
data={
|
||||||
|
"include_filters": "",
|
||||||
|
"fetch_backend": 'html_requests',
|
||||||
|
"ignore_text": "something to ignore",
|
||||||
|
"trigger_text": "something to trigger",
|
||||||
|
"url": test_url,
|
||||||
|
},
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
assert b"Updated watch." in res.data
|
||||||
|
wait_for_all_checks(client)
|
||||||
|
|
||||||
|
# The endpoint is a POST and accepts the form values to override the watch preview
|
||||||
|
import json
|
||||||
|
|
||||||
|
# DEFAULT OUTPUT WITHOUT ANYTHING UPDATED/CHANGED - SHOULD SEE THE WATCH DEFAULTS
|
||||||
|
res = client.post(
|
||||||
|
url_for("watch_get_preview_rendered", uuid=uuid)
|
||||||
|
)
|
||||||
|
default_return = json.loads(res.data.decode('utf-8'))
|
||||||
|
assert default_return.get('after_filter')
|
||||||
|
assert default_return.get('before_filter')
|
||||||
|
assert default_return.get('ignore_line_numbers') == [3] # "something to ignore" line 3
|
||||||
|
assert default_return.get('trigger_line_numbers') == [4] # "something to trigger" line 4
|
||||||
|
|
||||||
|
# SEND AN UPDATE AND WE SHOULD SEE THE OUTPUT CHANGE SO WE KNOW TO HIGHLIGHT NEW STUFF
|
||||||
|
res = client.post(
|
||||||
|
url_for("watch_get_preview_rendered", uuid=uuid),
|
||||||
|
data={
|
||||||
|
"include_filters": "",
|
||||||
|
"fetch_backend": 'html_requests',
|
||||||
|
"ignore_text": "sOckS", # Also be sure case insensitive works
|
||||||
|
"trigger_text": "AweSOme",
|
||||||
|
"url": test_url,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
reply = json.loads(res.data.decode('utf-8'))
|
||||||
|
assert reply.get('after_filter')
|
||||||
|
assert reply.get('before_filter')
|
||||||
|
assert reply.get('ignore_line_numbers') == [2] # Ignored - "socks" on line 2
|
||||||
|
assert reply.get('trigger_line_numbers') == [1] # Triggers "Awesome" in line 1
|
||||||
|
|
||||||
|
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
||||||
|
assert b'Deleted' in res.data
|
||||||
@@ -429,3 +429,24 @@ def test_global_send_test_notification(client, live_server, measure_memory_usage
|
|||||||
follow_redirects=True
|
follow_redirects=True
|
||||||
)
|
)
|
||||||
|
|
||||||
|
#2727 - be sure a test notification when there are zero watches works ( should all be deleted now)
|
||||||
|
|
||||||
|
os.unlink("test-datastore/notification.txt")
|
||||||
|
|
||||||
|
|
||||||
|
######### Test global/system settings
|
||||||
|
res = client.post(
|
||||||
|
url_for("ajax_callback_send_notification_test")+"?mode=global-settings",
|
||||||
|
data={"notification_urls": test_notification_url},
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
|
||||||
|
assert res.status_code != 400
|
||||||
|
assert res.status_code != 500
|
||||||
|
|
||||||
|
# Give apprise time to fire
|
||||||
|
time.sleep(4)
|
||||||
|
|
||||||
|
with open("test-datastore/notification.txt", 'r') as f:
|
||||||
|
x = f.read()
|
||||||
|
assert 'change detection is cool 网站监测 内容更新了' in x
|
||||||
|
|||||||
72
changedetectionio/tests/test_preview_endpoints.py
Normal file
72
changedetectionio/tests/test_preview_endpoints.py
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import time
|
||||||
|
from flask import url_for
|
||||||
|
from .util import set_original_response, set_modified_response, live_server_setup, wait_for_all_checks
|
||||||
|
|
||||||
|
|
||||||
|
# `subtractive_selectors` should still work in `source:` type requests
|
||||||
|
def test_fetch_pdf(client, live_server, measure_memory_usage):
|
||||||
|
import shutil
|
||||||
|
shutil.copy("tests/test.pdf", "test-datastore/endpoint-test.pdf")
|
||||||
|
|
||||||
|
live_server_setup(live_server)
|
||||||
|
test_url = url_for('test_pdf_endpoint', _external=True)
|
||||||
|
# Add our URL to the import page
|
||||||
|
res = client.post(
|
||||||
|
url_for("import_page"),
|
||||||
|
data={"urls": test_url},
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
|
||||||
|
assert b"1 Imported" in res.data
|
||||||
|
|
||||||
|
wait_for_all_checks(client)
|
||||||
|
|
||||||
|
res = client.get(
|
||||||
|
url_for("preview_page", uuid="first"),
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
|
||||||
|
# PDF header should not be there (it was converted to text)
|
||||||
|
assert b'PDF' not in res.data[:10]
|
||||||
|
assert b'hello world' in res.data
|
||||||
|
|
||||||
|
# So we know if the file changes in other ways
|
||||||
|
import hashlib
|
||||||
|
original_md5 = hashlib.md5(open("test-datastore/endpoint-test.pdf", 'rb').read()).hexdigest().upper()
|
||||||
|
# We should have one
|
||||||
|
assert len(original_md5) > 0
|
||||||
|
# And it's going to be in the document
|
||||||
|
assert b'Document checksum - ' + bytes(str(original_md5).encode('utf-8')) in res.data
|
||||||
|
|
||||||
|
shutil.copy("tests/test2.pdf", "test-datastore/endpoint-test.pdf")
|
||||||
|
changed_md5 = hashlib.md5(open("test-datastore/endpoint-test.pdf", 'rb').read()).hexdigest().upper()
|
||||||
|
res = client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
|
assert b'1 watches queued for rechecking.' in res.data
|
||||||
|
|
||||||
|
wait_for_all_checks(client)
|
||||||
|
|
||||||
|
# Now something should be ready, indicated by having a 'unviewed' class
|
||||||
|
res = client.get(url_for("index"))
|
||||||
|
assert b'unviewed' in res.data
|
||||||
|
|
||||||
|
# The original checksum should be not be here anymore (cdio adds it to the bottom of the text)
|
||||||
|
|
||||||
|
res = client.get(
|
||||||
|
url_for("preview_page", uuid="first"),
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
|
||||||
|
assert original_md5.encode('utf-8') not in res.data
|
||||||
|
assert changed_md5.encode('utf-8') in res.data
|
||||||
|
|
||||||
|
res = client.get(
|
||||||
|
url_for("diff_history_page", uuid="first"),
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
|
||||||
|
assert original_md5.encode('utf-8') in res.data
|
||||||
|
assert changed_md5.encode('utf-8') in res.data
|
||||||
|
|
||||||
|
assert b'here is a change' in res.data
|
||||||
@@ -3,7 +3,7 @@ import os
|
|||||||
import time
|
import time
|
||||||
|
|
||||||
from flask import url_for
|
from flask import url_for
|
||||||
from .util import live_server_setup, wait_for_all_checks, extract_UUID_from_client, wait_for_notification_endpoint_output
|
from .util import live_server_setup, wait_for_all_checks, wait_for_notification_endpoint_output
|
||||||
from ..notification import default_notification_format
|
from ..notification import default_notification_format
|
||||||
|
|
||||||
instock_props = [
|
instock_props = [
|
||||||
@@ -413,3 +413,31 @@ def test_data_sanity(client, live_server):
|
|||||||
res = client.get(
|
res = client.get(
|
||||||
url_for("edit_page", uuid="first"))
|
url_for("edit_page", uuid="first"))
|
||||||
assert test_url2.encode('utf-8') in res.data
|
assert test_url2.encode('utf-8') in res.data
|
||||||
|
|
||||||
|
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
||||||
|
assert b'Deleted' in res.data
|
||||||
|
|
||||||
|
# All examples should give a prive of 666.66
|
||||||
|
def test_special_prop_examples(client, live_server):
|
||||||
|
import glob
|
||||||
|
#live_server_setup(live_server)
|
||||||
|
|
||||||
|
test_url = url_for('test_endpoint', _external=True)
|
||||||
|
check_path = os.path.join(os.path.dirname(__file__), "itemprop_test_examples", "*.txt")
|
||||||
|
files = glob.glob(check_path)
|
||||||
|
assert files
|
||||||
|
for test_example_filename in files:
|
||||||
|
with open(test_example_filename, 'r') as example_f:
|
||||||
|
with open("test-datastore/endpoint-content.txt", "w") as test_f:
|
||||||
|
test_f.write(f"<html><body>{example_f.read()}</body></html>")
|
||||||
|
|
||||||
|
# Now fetch it and check the price worked
|
||||||
|
client.post(
|
||||||
|
url_for("form_quick_watch_add"),
|
||||||
|
data={"url": test_url, "tags": 'restock tests', 'processor': 'restock_diff'},
|
||||||
|
follow_redirects=True
|
||||||
|
)
|
||||||
|
wait_for_all_checks(client)
|
||||||
|
res = client.get(url_for("index"))
|
||||||
|
assert b'ception' not in res.data
|
||||||
|
assert b'155.55' in res.data
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
import time
|
import time
|
||||||
from flask import url_for
|
from flask import url_for
|
||||||
from . util import live_server_setup
|
from .util import live_server_setup, wait_for_all_checks
|
||||||
|
|
||||||
|
|
||||||
def set_original_ignore_response():
|
def set_original_ignore_response():
|
||||||
@@ -59,12 +59,9 @@ def test_trigger_functionality(client, live_server, measure_memory_usage):
|
|||||||
|
|
||||||
live_server_setup(live_server)
|
live_server_setup(live_server)
|
||||||
|
|
||||||
sleep_time_for_fetch_thread = 3
|
|
||||||
trigger_text = "Add to cart"
|
trigger_text = "Add to cart"
|
||||||
set_original_ignore_response()
|
set_original_ignore_response()
|
||||||
|
|
||||||
# Give the endpoint time to spin up
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
# Add our URL to the import page
|
# Add our URL to the import page
|
||||||
test_url = url_for('test_endpoint', _external=True)
|
test_url = url_for('test_endpoint', _external=True)
|
||||||
@@ -89,14 +86,14 @@ def test_trigger_functionality(client, live_server, measure_memory_usage):
|
|||||||
)
|
)
|
||||||
assert b"Updated watch." in res.data
|
assert b"Updated watch." in res.data
|
||||||
|
|
||||||
|
wait_for_all_checks(client)
|
||||||
# Check it saved
|
# Check it saved
|
||||||
res = client.get(
|
res = client.get(
|
||||||
url_for("edit_page", uuid="first"),
|
url_for("edit_page", uuid="first"),
|
||||||
)
|
)
|
||||||
assert bytes(trigger_text.encode('utf-8')) in res.data
|
assert bytes(trigger_text.encode('utf-8')) in res.data
|
||||||
|
|
||||||
# Give the thread time to pick it up
|
|
||||||
time.sleep(sleep_time_for_fetch_thread)
|
|
||||||
|
|
||||||
# so that we set the state to 'unviewed' after all the edits
|
# so that we set the state to 'unviewed' after all the edits
|
||||||
client.get(url_for("diff_history_page", uuid="first"))
|
client.get(url_for("diff_history_page", uuid="first"))
|
||||||
@@ -104,8 +101,7 @@ def test_trigger_functionality(client, live_server, measure_memory_usage):
|
|||||||
# Trigger a check
|
# Trigger a check
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
|
|
||||||
# Give the thread time to pick it up
|
wait_for_all_checks(client)
|
||||||
time.sleep(sleep_time_for_fetch_thread)
|
|
||||||
|
|
||||||
# It should report nothing found (no new 'unviewed' class)
|
# It should report nothing found (no new 'unviewed' class)
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
@@ -117,19 +113,17 @@ def test_trigger_functionality(client, live_server, measure_memory_usage):
|
|||||||
|
|
||||||
# Trigger a check
|
# Trigger a check
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
# Give the thread time to pick it up
|
wait_for_all_checks(client)
|
||||||
time.sleep(sleep_time_for_fetch_thread)
|
|
||||||
|
|
||||||
# It should report nothing found (no new 'unviewed' class)
|
# It should report nothing found (no new 'unviewed' class)
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
assert b'unviewed' not in res.data
|
assert b'unviewed' not in res.data
|
||||||
|
|
||||||
# Now set the content which contains the trigger text
|
# Now set the content which contains the trigger text
|
||||||
time.sleep(sleep_time_for_fetch_thread)
|
|
||||||
set_modified_with_trigger_text_response()
|
set_modified_with_trigger_text_response()
|
||||||
|
|
||||||
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
client.get(url_for("form_watch_checknow"), follow_redirects=True)
|
||||||
time.sleep(sleep_time_for_fetch_thread)
|
wait_for_all_checks(client)
|
||||||
res = client.get(url_for("index"))
|
res = client.get(url_for("index"))
|
||||||
assert b'unviewed' in res.data
|
assert b'unviewed' in res.data
|
||||||
|
|
||||||
@@ -142,4 +136,7 @@ def test_trigger_functionality(client, live_server, measure_memory_usage):
|
|||||||
res = client.get(url_for("preview_page", uuid="first"))
|
res = client.get(url_for("preview_page", uuid="first"))
|
||||||
|
|
||||||
# We should be able to see what we triggered on
|
# We should be able to see what we triggered on
|
||||||
assert b'<div class="triggered">Add to cart' in res.data
|
# The JS highlighter should tell us which lines (also used in the live-preview)
|
||||||
|
assert b'const triggered_line_numbers = [6]' in res.data
|
||||||
|
assert b'Add to cart' in res.data
|
||||||
|
|
||||||
|
|||||||
@@ -161,8 +161,8 @@ def test_check_xpath_text_function_utf8(client, live_server, measure_memory_usag
|
|||||||
follow_redirects=True
|
follow_redirects=True
|
||||||
)
|
)
|
||||||
|
|
||||||
assert b'<div class="">Stock Alert (UK): RPi CM4' in res.data
|
assert b'Stock Alert (UK): RPi CM4' in res.data
|
||||||
assert b'<div class="">Stock Alert (UK): Big monitor' in res.data
|
assert b'Stock Alert (UK): Big monitor' in res.data
|
||||||
|
|
||||||
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
res = client.get(url_for("form_delete", uuid="all"), follow_redirects=True)
|
||||||
assert b'Deleted' in res.data
|
assert b'Deleted' in res.data
|
||||||
|
|||||||
@@ -18,12 +18,13 @@ class TestDiffBuilder(unittest.TestCase):
|
|||||||
|
|
||||||
watch['last_viewed'] = 110
|
watch['last_viewed'] = 110
|
||||||
|
|
||||||
watch.save_history_text(contents=b"hello world", timestamp=100, snapshot_id=str(uuid_builder.uuid4()))
|
# Contents from the browser are always returned from the browser/requests/etc as str, str is basically UTF-16 in python
|
||||||
watch.save_history_text(contents=b"hello world", timestamp=105, snapshot_id=str(uuid_builder.uuid4()))
|
watch.save_history_text(contents="hello world", timestamp=100, snapshot_id=str(uuid_builder.uuid4()))
|
||||||
watch.save_history_text(contents=b"hello world", timestamp=109, snapshot_id=str(uuid_builder.uuid4()))
|
watch.save_history_text(contents="hello world", timestamp=105, snapshot_id=str(uuid_builder.uuid4()))
|
||||||
watch.save_history_text(contents=b"hello world", timestamp=112, snapshot_id=str(uuid_builder.uuid4()))
|
watch.save_history_text(contents="hello world", timestamp=109, snapshot_id=str(uuid_builder.uuid4()))
|
||||||
watch.save_history_text(contents=b"hello world", timestamp=115, snapshot_id=str(uuid_builder.uuid4()))
|
watch.save_history_text(contents="hello world", timestamp=112, snapshot_id=str(uuid_builder.uuid4()))
|
||||||
watch.save_history_text(contents=b"hello world", timestamp=117, snapshot_id=str(uuid_builder.uuid4()))
|
watch.save_history_text(contents="hello world", timestamp=115, snapshot_id=str(uuid_builder.uuid4()))
|
||||||
|
watch.save_history_text(contents="hello world", timestamp=117, snapshot_id=str(uuid_builder.uuid4()))
|
||||||
|
|
||||||
p = watch.get_next_snapshot_key_to_last_viewed
|
p = watch.get_next_snapshot_key_to_last_viewed
|
||||||
assert p == "112", "Correct last-viewed timestamp was detected"
|
assert p == "112", "Correct last-viewed timestamp was detected"
|
||||||
|
|||||||
@@ -81,7 +81,8 @@ class update_worker(threading.Thread):
|
|||||||
'watch_url': watch.get('url') if watch else None,
|
'watch_url': watch.get('url') if watch else None,
|
||||||
})
|
})
|
||||||
|
|
||||||
n_object.update(watch.extra_notification_token_values())
|
if watch:
|
||||||
|
n_object.update(watch.extra_notification_token_values())
|
||||||
|
|
||||||
logger.trace(f"Main rendered notification placeholders (diff_added etc) calculated in {time.time()-now:.3f}s")
|
logger.trace(f"Main rendered notification placeholders (diff_added etc) calculated in {time.time()-now:.3f}s")
|
||||||
logger.debug("Queued notification for sending")
|
logger.debug("Queued notification for sending")
|
||||||
@@ -260,9 +261,6 @@ class update_worker(threading.Thread):
|
|||||||
try:
|
try:
|
||||||
# Processor is what we are using for detecting the "Change"
|
# Processor is what we are using for detecting the "Change"
|
||||||
processor = watch.get('processor', 'text_json_diff')
|
processor = watch.get('processor', 'text_json_diff')
|
||||||
# Abort processing when the content was the same as the last fetch
|
|
||||||
skip_when_same_checksum = queued_item_data.item.get('skip_when_checksum_same')
|
|
||||||
|
|
||||||
|
|
||||||
# Init a new 'difference_detection_processor', first look in processors
|
# Init a new 'difference_detection_processor', first look in processors
|
||||||
processor_module_name = f"changedetectionio.processors.{processor}.processor"
|
processor_module_name = f"changedetectionio.processors.{processor}.processor"
|
||||||
@@ -278,16 +276,13 @@ class update_worker(threading.Thread):
|
|||||||
|
|
||||||
update_handler.call_browser()
|
update_handler.call_browser()
|
||||||
|
|
||||||
changed_detected, update_obj, contents, content_after_filters = update_handler.run_changedetection(
|
changed_detected, update_obj, contents = update_handler.run_changedetection(watch=watch)
|
||||||
watch=watch,
|
|
||||||
skip_when_checksum_same=skip_when_same_checksum,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Re #342
|
# Re #342
|
||||||
# In Python 3, all strings are sequences of Unicode characters. There is a bytes type that holds raw bytes.
|
# In Python 3, all strings are sequences of Unicode characters. There is a bytes type that holds raw bytes.
|
||||||
# We then convert/.decode('utf-8') for the notification etc
|
# We then convert/.decode('utf-8') for the notification etc
|
||||||
if not isinstance(contents, (bytes, bytearray)):
|
# if not isinstance(contents, (bytes, bytearray)):
|
||||||
raise Exception("Error - returned data from the fetch handler SHOULD be bytes")
|
# raise Exception("Error - returned data from the fetch handler SHOULD be bytes")
|
||||||
except PermissionError as e:
|
except PermissionError as e:
|
||||||
logger.critical(f"File permission error updating file, watch: {uuid}")
|
logger.critical(f"File permission error updating file, watch: {uuid}")
|
||||||
logger.critical(str(e))
|
logger.critical(str(e))
|
||||||
@@ -338,7 +333,8 @@ class update_worker(threading.Thread):
|
|||||||
elif e.status_code == 500:
|
elif e.status_code == 500:
|
||||||
err_text = "Error - 500 (Internal server error) received from the web site"
|
err_text = "Error - 500 (Internal server error) received from the web site"
|
||||||
else:
|
else:
|
||||||
err_text = "Error - Request returned a HTTP error code {}".format(str(e.status_code))
|
extra = ' (Access denied or blocked)' if str(e.status_code).startswith('4') else ''
|
||||||
|
err_text = f"Error - Request returned a HTTP error code {e.status_code}{extra}"
|
||||||
|
|
||||||
if e.screenshot:
|
if e.screenshot:
|
||||||
watch.save_screenshot(screenshot=e.screenshot, as_error=True)
|
watch.save_screenshot(screenshot=e.screenshot, as_error=True)
|
||||||
@@ -491,6 +487,8 @@ class update_worker(threading.Thread):
|
|||||||
if not self.datastore.data['watching'].get(uuid):
|
if not self.datastore.data['watching'].get(uuid):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
update_obj['content-type'] = update_handler.fetcher.get_all_headers().get('content-type', '').lower()
|
||||||
|
|
||||||
# Mark that we never had any failures
|
# Mark that we never had any failures
|
||||||
if not watch.get('ignore_status_codes'):
|
if not watch.get('ignore_status_codes'):
|
||||||
update_obj['consecutive_filter_failures'] = 0
|
update_obj['consecutive_filter_failures'] = 0
|
||||||
@@ -571,6 +569,12 @@ class update_worker(threading.Thread):
|
|||||||
except Exception as e:
|
except Exception as e:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
try:
|
||||||
|
watch.post_process()
|
||||||
|
except Exception as e:
|
||||||
|
logger.critical(e)
|
||||||
|
|
||||||
|
|
||||||
self.datastore.update_watch(uuid=uuid, update_obj={'fetch_time': round(time.time() - now, 3),
|
self.datastore.update_watch(uuid=uuid, update_obj={'fetch_time': round(time.time() - now, 3),
|
||||||
'last_checked': round(time.time()),
|
'last_checked': round(time.time()),
|
||||||
'check_count': count
|
'check_count': count
|
||||||
|
|||||||
@@ -58,6 +58,10 @@ services:
|
|||||||
#
|
#
|
||||||
# Absolute minimum seconds to recheck, overrides any watch minimum, change to 0 to disable
|
# Absolute minimum seconds to recheck, overrides any watch minimum, change to 0 to disable
|
||||||
# - MINIMUM_SECONDS_RECHECK_TIME=3
|
# - MINIMUM_SECONDS_RECHECK_TIME=3
|
||||||
|
#
|
||||||
|
# If you want to watch local files file:///path/to/file.txt (careful! security implications!)
|
||||||
|
# - ALLOW_FILE_URI=False
|
||||||
|
|
||||||
# Comment out ports: when using behind a reverse proxy , enable networks: etc.
|
# Comment out ports: when using behind a reverse proxy , enable networks: etc.
|
||||||
ports:
|
ports:
|
||||||
- 5000:5000
|
- 5000:5000
|
||||||
|
|||||||
@@ -93,3 +93,5 @@ babel
|
|||||||
|
|
||||||
# Needed for > 3.10, https://github.com/microsoft/playwright-python/issues/2096
|
# Needed for > 3.10, https://github.com/microsoft/playwright-python/issues/2096
|
||||||
greenlet >= 3.0.3
|
greenlet >= 3.0.3
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user