mirror of
https://github.com/dgtlmoon/changedetection.io.git
synced 2026-05-29 21:11:50 +00:00
3d14df6a11
Multi-language / Translations Support (#3696) - Complete internationalization system implemented - Support for 7 languages: Czech (cs), German (de), French (fr), Italian (it), Korean (ko), Chinese Simplified (zh), Chinese Traditional (zh_TW) - Language selector with localized flags and theming - Flash message translations - Multiple translation fixes and improvements across all languages - Language setting preserved across redirects Pluggable Content Fetchers (#3653) - New architecture for extensible content fetcher system - Allows custom fetcher implementations Image / Screenshot Comparison Processor (#3680) - New processor for visual change detection (disabled for this release) - Supporting CSS/JS infrastructure added UI Improvements Design & Layout - Auto-generated tag color schemes - Simplified login form styling - Removed hard-coded CSS, moved to SCSS variables - Tag UI cleanup and improvements - Automatic tab wrapper functionality - Menu refactoring for better organization - Cleanup of offset settings - Hide sticky tabs on narrow viewports - Improved responsive layout (#3702) User Experience - Modal alerts/confirmations on delete/clear operations (#3693, #3598, #3382) - Auto-add https:// to URLs in quickwatch form if not present - Better redirect handling on login (#3699) - 'Recheck all' now returns to correct group/tag (#3673) - Language set redirect keeps hash fragment - More friendly human-readable text throughout UI Performance & Reliability Scheduler & Processing - Soft delays instead of blocking time.sleep() calls (#3710) - More resilient handling of same UUID being processed (#3700) - Better Puppeteer timeout handling - Improved Puppeteer shutdown/cleanup (#3692) - Requests cleanup now properly async History & Rendering - Faster server-side "difference" rendering on History page (#3442) - Show ignored/triggered rows in history - API: Retry watch data if watch dict changed (more reliable) API Improvements - Watch get endpoint: retry mechanism for changed watch data - WatchHistoryDiff API endpoint includes extra format args (#3703) Testing Improvements - Replace time.sleep with wait_for_notification_endpoint_output (#3716) - Test for mode switching (#3701) - Test for #3720 added (#3725) - Extract-text difference test fixes - Improved dev workflow Bug Fixes - Notification error text output (#3672, #3669, #3280) - HTML validation fixes (#3704) - Template discovery path fixes - Notification debug log now uses system locale for dates/times - Puppeteer spelling mistake in log output - Recalculation on anchor change - Queue bubble update disabled temporarily Dependency Updates - beautifulsoup4 updated (#3724) - psutil 7.1.0 → 7.2.1 (#3723) - python-engineio ~=4.12.3 → ~=4.13.0 (#3707) - python-socketio ~=5.14.3 → ~=5.16.0 (#3706) - flask-socketio ~=5.5.1 → ~=5.6.0 (#3691) - brotli ~=1.1 → ~=1.2 (#3687) - lxml updated (#3590) - pytest ~=7.2 → ~=9.0 (#3676) - jsonschema ~=4.0 → ~=4.25 (#3618) - pluggy ~=1.5 → ~=1.6 (#3616) - cryptography 44.0.1 → 46.0.3 (security) (#3589) Documentation - README updated with viewport size setup information Development Infrastructure - Dev container only built on dev branch - Improved dev workflow tooling
221 lines
8.4 KiB
Python
221 lines
8.4 KiB
Python
#!/usr/bin/env python3
|
|
|
|
import time
|
|
from flask import url_for
|
|
from .util import live_server_setup, extract_UUID_from_client, wait_for_all_checks
|
|
import os
|
|
|
|
|
|
def set_response_with_ldjson(datastore_path):
|
|
test_return_data = """<html>
|
|
<body>
|
|
Some initial text<br>
|
|
<p>Which is across multiple lines</p>
|
|
<br>
|
|
So let's see what happens. <br>
|
|
<div class="sametext">Some text thats the same</div>
|
|
<div class="changetext">Some text that will change</div>
|
|
<script type="application/ld+json">
|
|
{
|
|
"@context":"https://schema.org/",
|
|
"@type":"Product",
|
|
"@id":"https://www.some-virtual-phone-shop.com/celular-iphone-14/p",
|
|
"name":"Celular Iphone 14 Pro Max 256Gb E Sim A16 Bionic",
|
|
"brand":{
|
|
"@type":"Brand",
|
|
"name":"APPLE"
|
|
},
|
|
"image":"https://www.some-virtual-phone-shop.com/15509426/image.jpg",
|
|
"description":"You dont need it",
|
|
"mpn":"111111",
|
|
"sku":"22222",
|
|
"Offers":{
|
|
"@type":"AggregateOffer",
|
|
"lowPrice":8097000,
|
|
"highPrice":8099900,
|
|
"priceCurrency":"COP",
|
|
"offers":[
|
|
{
|
|
"@type":"Offer",
|
|
"price":8097000,
|
|
"priceCurrency":"COP",
|
|
"availability":"http://schema.org/InStock",
|
|
"sku":"102375961",
|
|
"itemCondition":"http://schema.org/NewCondition",
|
|
"seller":{
|
|
"@type":"Organization",
|
|
"name":"ajax"
|
|
}
|
|
}
|
|
],
|
|
"offerCount":1
|
|
}
|
|
}
|
|
</script>
|
|
</body>
|
|
</html>
|
|
"""
|
|
|
|
with open(os.path.join(datastore_path, "endpoint-content.txt"), "w") as f:
|
|
f.write(test_return_data)
|
|
return None
|
|
|
|
def set_response_without_ldjson(datastore_path):
|
|
test_return_data = """<html>
|
|
<body>
|
|
Some initial text<br>
|
|
<p>Which is across multiple lines</p>
|
|
<br>
|
|
So let's see what happens. <br>
|
|
<div class="sametext">Some text thats the same</div>
|
|
<div class="changetext">Some text that will change</div>
|
|
</body>
|
|
</html>
|
|
"""
|
|
|
|
with open(os.path.join(datastore_path, "endpoint-content.txt"), "w") as f:
|
|
f.write(test_return_data)
|
|
return None
|
|
|
|
# def test_setup(client, live_server, measure_memory_usage, datastore_path):
|
|
# live_server_setup(live_server) # Setup on conftest per function
|
|
|
|
# actually only really used by the distll.io importer, but could be handy too
|
|
def test_check_ldjson_price_autodetect(client, live_server, measure_memory_usage, datastore_path):
|
|
|
|
set_response_with_ldjson(datastore_path=datastore_path)
|
|
|
|
# Add our URL to the import page
|
|
test_url = url_for('test_endpoint', _external=True)
|
|
uuid = client.application.config.get('DATASTORE').add_watch(url=test_url)
|
|
client.get(url_for("ui.form_watch_checknow"), follow_redirects=True)
|
|
wait_for_all_checks(client)
|
|
|
|
# Should get a notice that it's available
|
|
res = client.get(url_for("watchlist.index"))
|
|
assert b'ldjson-price-track-offer' in res.data
|
|
|
|
# Accept it
|
|
client.get(url_for('price_data_follower.accept', uuid=uuid, follow_redirects=True))
|
|
client.get(url_for("ui.form_watch_checknow"), follow_redirects=True)
|
|
wait_for_all_checks(client)
|
|
# Offer should be gone
|
|
res = client.get(url_for("watchlist.index"))
|
|
assert b'Embedded price data' not in res.data
|
|
assert b'processor-badge-restock_diff' in res.data
|
|
|
|
# and last snapshop (via API) should be just the price
|
|
api_key = live_server.app.config['DATASTORE'].data['settings']['application'].get('api_access_token')
|
|
res = client.get(
|
|
url_for("watchsinglehistory", uuid=uuid, timestamp='latest'),
|
|
headers={'x-api-key': api_key},
|
|
)
|
|
|
|
assert b'8097000' in res.data
|
|
|
|
# And not this cause its not the ld-json
|
|
assert b"So let's see what happens" not in res.data
|
|
|
|
client.get(url_for("ui.form_delete", uuid="all"), follow_redirects=True)
|
|
|
|
##########################################################################################
|
|
# And we shouldnt see the offer
|
|
set_response_without_ldjson(datastore_path=datastore_path)
|
|
|
|
# Add our URL to the import page
|
|
test_url = url_for('test_endpoint', _external=True)
|
|
uuid = client.application.config.get('DATASTORE').add_watch(url=test_url)
|
|
client.get(url_for("ui.form_watch_checknow"), follow_redirects=True)
|
|
wait_for_all_checks(client)
|
|
res = client.get(url_for("watchlist.index"))
|
|
assert b'ldjson-price-track-offer' not in res.data
|
|
|
|
##########################################################################################
|
|
client.get(url_for("ui.form_delete", uuid="all"), follow_redirects=True)
|
|
|
|
|
|
def _test_runner_check_bad_format_ignored(live_server, client, has_ldjson_price_data):
|
|
|
|
test_url = url_for('test_endpoint', _external=True)
|
|
uuid = client.application.config.get('DATASTORE').add_watch(url=test_url)
|
|
client.get(url_for("ui.form_watch_checknow"), follow_redirects=True)
|
|
wait_for_all_checks(client)
|
|
|
|
for k,v in client.application.config.get('DATASTORE').data['watching'].items():
|
|
assert v.get('last_error') == False
|
|
assert v.get('has_ldjson_price_data') == has_ldjson_price_data, f"Detected LDJSON data? should be {has_ldjson_price_data}"
|
|
|
|
|
|
##########################################################################################
|
|
client.get(url_for("ui.form_delete", uuid="all"), follow_redirects=True)
|
|
|
|
|
|
def test_bad_ldjson_is_correctly_ignored(client, live_server, measure_memory_usage, datastore_path):
|
|
|
|
test_return_data = """
|
|
<html>
|
|
<head>
|
|
<script type="application/ld+json">
|
|
{
|
|
"@context": "http://schema.org",
|
|
"@type": ["Product", "SubType"],
|
|
"name": "My test product",
|
|
"description": "",
|
|
"offers": {
|
|
"note" : "You can see the case-insensitive OffERS key, it should work",
|
|
"@type": "Offer",
|
|
"offeredBy": {
|
|
"@type": "Organization",
|
|
"name":"Person",
|
|
"telephone":"+1 999 999 999"
|
|
},
|
|
"price": "1",
|
|
"priceCurrency": "EUR",
|
|
"url": "/some/url"
|
|
}
|
|
}
|
|
</script>
|
|
</head>
|
|
<body>
|
|
<div class="yes">Some extra stuff</div>
|
|
</body></html>
|
|
"""
|
|
with open(os.path.join(datastore_path, "endpoint-content.txt"), "w") as f:
|
|
f.write(test_return_data)
|
|
|
|
_test_runner_check_bad_format_ignored(live_server=live_server, client=client, has_ldjson_price_data=True)
|
|
|
|
# This is OK that it offers a suggestion in this case, the processor will let them know more about something wrong
|
|
|
|
# test_return_data = """
|
|
# <html>
|
|
# <head>
|
|
# <script type="application/ld+json">
|
|
# {
|
|
# "@context": "http://schema.org",
|
|
# "@type": ["Product", "SubType"],
|
|
# "name": "My test product",
|
|
# "description": "",
|
|
# "BrokenOffers": {
|
|
# "@type": "Offer",
|
|
# "offeredBy": {
|
|
# "@type": "Organization",
|
|
# "name":"Person",
|
|
# "telephone":"+1 999 999 999"
|
|
# },
|
|
# "price": "1",
|
|
# "priceCurrency": "EUR",
|
|
# "url": "/some/url"
|
|
# }
|
|
# }
|
|
# </script>
|
|
# </head>
|
|
# <body>
|
|
# <div class="yes">Some extra stuff</div>
|
|
# </body></html>
|
|
# """
|
|
# with open(os.path.join(datastore_path, "endpoint-content.txt"), "w") as f:
|
|
# f.write(test_return_data)
|
|
#
|
|
# _test_runner_check_bad_format_ignored(live_server=live_server, client=client, has_ldjson_price_data=False)
|