Skip to content

Commit

Permalink
Use g and current_app proxies
Browse files Browse the repository at this point in the history
This should make the code a little cleaner to use these proxy objects.
  • Loading branch information
kruton committed Nov 27, 2024
1 parent 10068e1 commit 25408a0
Show file tree
Hide file tree
Showing 12 changed files with 55 additions and 62 deletions.
10 changes: 5 additions & 5 deletions changedetectionio/__main__.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,19 +19,19 @@

# Parent wrapper or OS sends us a SIGTERM/SIGINT, do everything required for a clean shutdown
class SigShutdownHandler(object):
def __init__(self, app):
def __init__(self, app: Flask, datastore: store.ChangeDetectionStore):
self.app = app
self.datastore = datastore
signal.signal(signal.SIGTERM, lambda _signum, _frame: self._signal_handler("SIGTERM"))
signal.signal(signal.SIGINT, lambda _signum, _frame: self._signal_handler("SIGINT"))

def _signal_handler(self, signame):
logger.critical(f'Shutdown: Got Signal - {signame}, Saving DB to disk and calling shutdown')
datastore = self.app.config["DATASTORE"]
datastore.sync_to_json()
self.datastore.sync_to_json()
logger.success('Sync JSON to disk complete.')
# This will throw a SystemExit exception, because eventlet.wsgi.server doesn't know how to deal with it.
# Solution: move to gevent or other server in the future (#2014)
datastore.stop_thread = True
self.datastore.stop_thread = True
self.app.config.exit.set()
sys.exit(0)

Expand Down Expand Up @@ -136,7 +136,7 @@ def create_application() -> Flask:

app = changedetection_app(app_config, datastore)

sigshutdown_handler = SigShutdownHandler(app)
sigshutdown_handler = SigShutdownHandler(app, datastore)

# Go into cleanup mode
if do_cleanup:
Expand Down
41 changes: 17 additions & 24 deletions changedetectionio/flask_app.py
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
from flask import (
Flask,
abort,
current_app,
flash,
g,
make_response,
Expand Down Expand Up @@ -133,17 +134,17 @@ def decorated_view(*args, **kwargs):
has_password_enabled = g.datastore.data['settings']['application'].get('password') or os.getenv("SALTED_PASS", False)

# Permitted
if request.endpoint == 'static_content' and request.view_args['group'] == 'styles':
if request.endpoint == 'static_content' and request.view_args and request.view_args['group'] == 'styles':
return func(*args, **kwargs)
# Permitted
elif request.endpoint == 'diff_history_page' and g.datastore.data['settings']['application'].get('shared_diff_access'):
return func(*args, **kwargs)
elif request.method in flask_login.config.EXEMPT_METHODS:
return func(*args, **kwargs)
elif g.app.config.get('LOGIN_DISABLED'):
elif current_app.config.get('LOGIN_DISABLED'):
return func(*args, **kwargs)
elif has_password_enabled and not current_user.is_authenticated:
return g.app.login_manager.unauthorized()
return current_app.login_manager.unauthorized()

return func(*args, **kwargs)

Expand All @@ -165,7 +166,8 @@ def changedetection_app(config, datastore):

# Stop browser caching of assets
app.config['SEND_FILE_MAX_AGE_DEFAULT'] = 0
app.config.exit = Event()
exit_event = Event()
app.config.exit = exit_event

app.config['NEW_VERSION_AVAILABLE'] = False

Expand All @@ -182,8 +184,6 @@ def changedetection_app(config, datastore):

app.config["notification_debug_log"] = []

app.config['DATASTORE'] = datastore

login_manager = flask_login.LoginManager(app)
login_manager.login_view = 'login'
app.secret_key = init_app_secret(config['datastore_path'])
Expand Down Expand Up @@ -322,7 +322,6 @@ def login():

@app.before_request
def remember_app_and_datastore():
g.app = app
g.datastore = datastore

@app.before_request
Expand Down Expand Up @@ -1630,25 +1629,23 @@ def highlight_submit_ignore_url():


# @todo handle ctrl break
threading.Thread(target=ticker_thread_check_time_launch_checks, args=(app,)).start()
threading.Thread(target=notification_runner, args=(app,)).start()
threading.Thread(target=ticker_thread_check_time_launch_checks, kwargs={'app': app, 'datastore': datastore, 'exit_event': exit_event}).start()
threading.Thread(target=notification_runner, kwargs={'app': app, 'datastore': datastore, 'exit_event': exit_event}).start()

# Check for new release version, but not when running in test/build or pytest
if not os.getenv("GITHUB_REF", False) and not strtobool(os.getenv('DISABLE_VERSION_CHECK', 'no')):
threading.Thread(target=check_for_new_version, args=(app,)).start()
threading.Thread(target=check_for_new_version, kwargs={'app': app, 'datastore': datastore, 'exit_event': exit_event}).start()

return app


# Check for new version and anonymous stats
def check_for_new_version(app, url="https://changedetection.io/check-ver.php", delay_time=86400):
def check_for_new_version(*, app, datastore, exit_event, url="https://changedetection.io/check-ver.php", delay_time=86400):
import requests
import urllib3
urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning)

datastore = app.config["DATASTORE"]

while not app.config.exit.is_set():
while not exit_event.is_set():
try:
r = requests.post(url,
data={'version': __version__,
Expand All @@ -1667,13 +1664,13 @@ def check_for_new_version(app, url="https://changedetection.io/check-ver.php", d
pass

# Check daily
app.config.exit.wait(delay_time)
exit_event.wait(delay_time)


def notification_runner(app):
def notification_runner(*, app, datastore, exit_event):
from datetime import datetime
import json
while not app.config.exit.is_set():
while not exit_event.is_set():
try:
# At the moment only one thread runs (single runner)
n_object = notification_q.get(block=False)
Expand All @@ -1687,8 +1684,6 @@ def notification_runner(app):

notification_debug_log = app.config["notification_debug_log"]

datastore = app.config["DATASTORE"]

try:
from changedetectionio import notification
# Fallback to system config if not set
Expand Down Expand Up @@ -1720,12 +1715,10 @@ def notification_runner(app):
notification_debug_log = notification_debug_log[-100:]

# Threaded runner, look for new watches to feed into the Queue.
def ticker_thread_check_time_launch_checks(app):
def ticker_thread_check_time_launch_checks(*, app, datastore, exit_event):
import random
from changedetectionio import update_worker

datastore = app.config["DATASTORE"]

proxy_last_called_time = {}

recheck_time_minimum_seconds = int(os.getenv('MINIMUM_SECONDS_RECHECK_TIME', 3))
Expand All @@ -1739,7 +1732,7 @@ def ticker_thread_check_time_launch_checks(app):
running_update_threads.append(new_worker)
new_worker.start()

while not app.config.exit.is_set():
while not exit_event.is_set():

# Get a list of watches by UUID that are currently fetching data
running_uuids = []
Expand Down Expand Up @@ -1835,4 +1828,4 @@ def ticker_thread_check_time_launch_checks(app):
time.sleep(1)

# Should be low so we can break this out in testing
app.config.exit.wait(1)
exit_event.wait(1)
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import os
from flask import url_for
from flask import url_for, g
from ..util import live_server_setup, wait_for_all_checks, extract_UUID_from_client


Expand Down Expand Up @@ -35,7 +35,7 @@ def test_execute_custom_js(client, live_server, measure_memory_usage):
wait_for_all_checks(client)

uuid = extract_UUID_from_client(client)
assert live_server.app.config['DATASTORE'].data['watching'][uuid].history_n >= 1, "Watch history had atleast 1 (everything fetched OK)"
assert g.datastore.data['watching'][uuid].history_n >= 1, "Watch history had atleast 1 (everything fetched OK)"

assert b"This text should be removed" not in res.data

Expand All @@ -53,4 +53,4 @@ def test_execute_custom_js(client, live_server, measure_memory_usage):
client.get(
url_for("form_delete", uuid="all"),
follow_redirects=True
)
)
4 changes: 2 additions & 2 deletions changedetectionio/tests/proxy_list/test_noproxy.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
#!/usr/bin/env python3

import time
from flask import url_for
from flask import url_for, g
from ..util import live_server_setup, wait_for_all_checks, extract_UUID_from_client


Expand Down Expand Up @@ -73,5 +73,5 @@ def test_noproxy_option(client, live_server, measure_memory_usage):

# Prove that it actually checked

assert live_server.app.config['DATASTORE'].data['watching'][uuid]['last_checked'] != 0
assert g.datastore.data['watching'][uuid]['last_checked'] != 0

4 changes: 2 additions & 2 deletions changedetectionio/tests/test_automatic_follow_ldjson_price.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
#!/usr/bin/env python3

import time
from flask import url_for
from flask import url_for, g
from .util import live_server_setup, extract_UUID_from_client, extract_api_key_from_UI, wait_for_all_checks


Expand Down Expand Up @@ -154,7 +154,7 @@ def _test_runner_check_bad_format_ignored(live_server, client, has_ldjson_price_
assert b"1 Imported" in res.data
wait_for_all_checks(client)

for k,v in client.application.config.get('DATASTORE').data['watching'].items():
for k,v in g.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}"

Expand Down
4 changes: 2 additions & 2 deletions changedetectionio/tests/test_encoding.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
# coding=utf-8

import time
from flask import url_for
from flask import url_for, g
from .util import live_server_setup, wait_for_all_checks, extract_UUID_from_client
import pytest

Expand Down Expand Up @@ -41,7 +41,7 @@ def test_check_encoding_detection(client, live_server, measure_memory_usage):

# Content type recording worked
uuid = extract_UUID_from_client(client)
assert live_server.app.config['DATASTORE'].data['watching'][uuid]['content-type'] == "text/html"
assert g.datastore.data['watching'][uuid]['content-type'] == "text/html"

res = client.get(
url_for("preview_page", uuid="first"),
Expand Down
12 changes: 6 additions & 6 deletions changedetectionio/tests/test_filter_failure_notification.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import os
import time
from loguru import logger
from flask import url_for
from flask import url_for, g
from .util import set_original_response, live_server_setup, extract_UUID_from_client, wait_for_all_checks, \
wait_for_notification_endpoint_output
from changedetectionio.model import App
Expand Down Expand Up @@ -53,7 +53,7 @@ def run_filter_test(client, live_server, content_filter):

uuid = extract_UUID_from_client(client)

assert live_server.app.config['DATASTORE'].data['watching'][uuid]['consecutive_filter_failures'] == 0, "No filter = No filter failure"
assert g.datastore.data['watching'][uuid]['consecutive_filter_failures'] == 0, "No filter = No filter failure"

watch_data = {"notification_urls": notification_url,
"notification_title": "New ChangeDetection.io Notification - {{watch_url}}",
Expand Down Expand Up @@ -86,7 +86,7 @@ def run_filter_test(client, live_server, content_filter):
)
assert b"Updated watch." in res.data
wait_for_all_checks(client)
assert live_server.app.config['DATASTORE'].data['watching'][uuid]['consecutive_filter_failures'] == 0, "No filter = No filter failure"
assert g.datastore.data['watching'][uuid]['consecutive_filter_failures'] == 0, "No filter = No filter failure"

# Now add a filter, because recheck hours == 5, ONLY pressing of the [edit] or [recheck all] should trigger
watch_data['include_filters'] = content_filter
Expand All @@ -103,12 +103,12 @@ def run_filter_test(client, live_server, content_filter):
assert not os.path.isfile("test-datastore/notification.txt")

# Hitting [save] would have triggered a recheck, and we have a filter, so this would be ONE failure
assert live_server.app.config['DATASTORE'].data['watching'][uuid]['consecutive_filter_failures'] == 1, "Should have been checked once"
assert g.datastore.data['watching'][uuid]['consecutive_filter_failures'] == 1, "Should have been checked once"

# recheck it up to just before the threshold, including the fact that in the previous POST it would have rechecked (and incremented)
# Add 4 more checks
checked = 0
ATTEMPT_THRESHOLD_SETTING = live_server.app.config['DATASTORE'].data['settings']['application'].get('filter_failure_notification_threshold_attempts', 0)
ATTEMPT_THRESHOLD_SETTING = g.datastore.data['settings']['application'].get('filter_failure_notification_threshold_attempts', 0)
for i in range(0, ATTEMPT_THRESHOLD_SETTING - 2):
checked += 1
client.get(url_for("form_watch_checknow"), follow_redirects=True)
Expand All @@ -118,7 +118,7 @@ def run_filter_test(client, live_server, content_filter):
assert not os.path.isfile("test-datastore/notification.txt")
time.sleep(1)

assert live_server.app.config['DATASTORE'].data['watching'][uuid]['consecutive_filter_failures'] == 5
assert g.datastore.data['watching'][uuid]['consecutive_filter_failures'] == 5

time.sleep(2)
# One more check should trigger the _FILTER_FAILURE_THRESHOLD_ATTEMPTS_DEFAULT threshold
Expand Down
12 changes: 6 additions & 6 deletions changedetectionio/tests/test_history_consistency.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
import os
import json
import logging
from flask import url_for
from flask import url_for, g
from .util import live_server_setup, wait_for_all_checks
from urllib.parse import urlparse, parse_qs

Expand Down Expand Up @@ -38,7 +38,7 @@ def test_consistent_history(client, live_server, measure_memory_usage):

time.sleep(2)

json_db_file = os.path.join(live_server.app.config['DATASTORE'].datastore_path, 'url-watches.json')
json_db_file = os.path.join(g.datastore.datastore_path, 'url-watches.json')

json_obj = None
with open(json_db_file, 'r') as f:
Expand All @@ -49,7 +49,7 @@ def test_consistent_history(client, live_server, measure_memory_usage):

# each one should have a history.txt containing just one line
for w in json_obj['watching'].keys():
history_txt_index_file = os.path.join(live_server.app.config['DATASTORE'].datastore_path, w, 'history.txt')
history_txt_index_file = os.path.join(g.datastore.datastore_path, w, 'history.txt')
assert os.path.isfile(history_txt_index_file), f"History.txt should exist where I expect it at {history_txt_index_file}"

# Same like in model.Watch
Expand All @@ -58,13 +58,13 @@ def test_consistent_history(client, live_server, measure_memory_usage):
assert len(tmp_history) == 1, "History.txt should contain 1 line"

# Should be two files,. the history.txt , and the snapshot.txt
files_in_watch_dir = os.listdir(os.path.join(live_server.app.config['DATASTORE'].datastore_path,
files_in_watch_dir = os.listdir(os.path.join(g.datastore.datastore_path,
w))
# Find the snapshot one
for fname in files_in_watch_dir:
if fname != 'history.txt' and 'html' not in fname:
# contents should match what we requested as content returned from the test url
with open(os.path.join(live_server.app.config['DATASTORE'].datastore_path, w, fname), 'r') as snapshot_f:
with open(os.path.join(g.datastore.datastore_path, w, fname), 'r') as snapshot_f:
contents = snapshot_f.read()
watch_url = json_obj['watching'][w]['url']
u = urlparse(watch_url)
Expand All @@ -76,6 +76,6 @@ def test_consistent_history(client, live_server, measure_memory_usage):
assert len(files_in_watch_dir) == 3, "Should be just three files in the dir, html.br snapshot, history.txt and the extracted text snapshot"


json_db_file = os.path.join(live_server.app.config['DATASTORE'].datastore_path, 'url-watches.json')
json_db_file = os.path.join(g.datastore.datastore_path, 'url-watches.json')
with open(json_db_file, 'r') as f:
assert '"default"' not in f.read(), "'default' probably shouldnt be here, it came from when the 'default' Watch vars were accidently being saved"
6 changes: 3 additions & 3 deletions changedetectionio/tests/test_import.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
import os
import time

from flask import url_for
from flask import url_for, g

from .util import live_server_setup, wait_for_all_checks

Expand Down Expand Up @@ -163,7 +163,7 @@ def test_import_custom_xlsx(client, live_server, measure_memory_usage):
assert b'City news results' in res.data

# Just find one to check over
for uuid, watch in live_server.app.config['DATASTORE'].data['watching'].items():
for uuid, watch in g.datastore.data['watching'].items():
if watch.get('title') == 'Somesite results ABC':
filters = watch.get('include_filters')
assert filters[0] == '/html[1]/body[1]/div[4]/div[1]/div[1]/div[1]||//*[@id=\'content\']/div[3]/div[1]/div[1]||//*[@id=\'content\']/div[1]'
Expand Down Expand Up @@ -201,7 +201,7 @@ def test_import_watchete_xlsx(client, live_server, measure_memory_usage):
assert b'City news results' in res.data

# Just find one to check over
for uuid, watch in live_server.app.config['DATASTORE'].data['watching'].items():
for uuid, watch in g.datastore.data['watching'].items():
if watch.get('title') == 'Somesite results ABC':
filters = watch.get('include_filters')
assert filters[0] == '/html[1]/body[1]/div[4]/div[1]/div[1]/div[1]||//*[@id=\'content\']/div[3]/div[1]/div[1]||//*[@id=\'content\']/div[1]'
Expand Down
Loading

0 comments on commit 25408a0

Please sign in to comment.