automatyzacje-comit1

This commit is contained in:
Mateusz Gruszczyński
2026-05-07 07:24:16 +02:00
parent 2691442fc1
commit 440b187c39
6 changed files with 128 additions and 25 deletions

View File

@@ -326,9 +326,11 @@ def cleanup_summary() -> dict:
"jobs_total": _table_count("jobs"), "jobs_total": _table_count("jobs"),
"jobs_clearable": _table_count("jobs", "WHERE status NOT IN ('pending', 'running')"), "jobs_clearable": _table_count("jobs", "WHERE status NOT IN ('pending', 'running')"),
"smart_queue_history_total": _table_count("smart_queue_history"), "smart_queue_history_total": _table_count("smart_queue_history"),
"automation_history_total": _table_count("automation_history"),
"retention_days": { "retention_days": {
"jobs": JOBS_RETENTION_DAYS, "jobs": JOBS_RETENTION_DAYS,
"smart_queue_history": SMART_QUEUE_HISTORY_RETENTION_DAYS, "smart_queue_history": SMART_QUEUE_HISTORY_RETENTION_DAYS,
"automation_history": SMART_QUEUE_HISTORY_RETENTION_DAYS,
}, },
"database": _db_size(), "database": _db_size(),
} }
@@ -731,6 +733,19 @@ def cleanup_smart_queue():
return ok({"deleted": deleted, "cleanup": cleanup_summary()}) return ok({"deleted": deleted, "cleanup": cleanup_summary()})
@bp.post("/cleanup/automations")
def cleanup_automations():
with connect() as conn:
exists = conn.execute("SELECT name FROM sqlite_master WHERE type='table' AND name='automation_history'").fetchone()
if not exists:
deleted = 0
else:
# Note: Cleanup panel removes only automation logs, not saved automation rules.
cur = conn.execute("DELETE FROM automation_history")
deleted = int(cur.rowcount or 0)
return ok({"deleted": deleted, "cleanup": cleanup_summary()})
@bp.post("/cleanup/all") @bp.post("/cleanup/all")
def cleanup_all(): def cleanup_all():
deleted_jobs = clear_jobs() deleted_jobs = clear_jobs()
@@ -741,7 +756,13 @@ def cleanup_all():
else: else:
cur = conn.execute("DELETE FROM smart_queue_history") cur = conn.execute("DELETE FROM smart_queue_history")
deleted_smart = int(cur.rowcount or 0) deleted_smart = int(cur.rowcount or 0)
return ok({"deleted": {"jobs": deleted_jobs, "smart_queue_history": deleted_smart}, "cleanup": cleanup_summary()}) exists_auto = conn.execute("SELECT name FROM sqlite_master WHERE type='table' AND name='automation_history'").fetchone()
if not exists_auto:
deleted_auto = 0
else:
cur = conn.execute("DELETE FROM automation_history")
deleted_auto = int(cur.rowcount or 0)
return ok({"deleted": {"jobs": deleted_jobs, "smart_queue_history": deleted_smart, "automation_history": deleted_auto}, "cleanup": cleanup_summary()})
@bp.post("/jobs/<job_id>/cancel") @bp.post("/jobs/<job_id>/cancel")
@@ -1062,3 +1083,17 @@ def automations_check():
return ok({'result': automation_rules.check(profile, force=True), 'history': automation_rules.list_history(profile['id'])}) return ok({'result': automation_rules.check(profile, force=True), 'history': automation_rules.list_history(profile['id'])})
except Exception as exc: except Exception as exc:
return jsonify({'ok': False, 'error': str(exc)}), 500 return jsonify({'ok': False, 'error': str(exc)}), 500
@bp.delete('/automations/history')
def automations_history_clear():
from ..services import automation_rules
profile = preferences.active_profile()
if not profile:
return jsonify({'ok': False, 'error': 'No profile'}), 400
try:
# Note: Clear only automation execution logs; rules and cooldown state stay unchanged.
deleted = automation_rules.clear_history(profile['id'])
return ok({'deleted': deleted, 'history': automation_rules.list_history(profile['id']), 'cleanup': cleanup_summary()})
except Exception as exc:
return jsonify({'ok': False, 'error': str(exc)}), 500

View File

@@ -94,6 +94,14 @@ def list_history(profile_id: int, user_id: int | None = None, limit: int = 30) -
return conn.execute('SELECT * FROM automation_history WHERE user_id=? AND profile_id=? ORDER BY created_at DESC LIMIT ?', (user_id, profile_id, max(1, min(int(limit or 30), 100)))).fetchall() return conn.execute('SELECT * FROM automation_history WHERE user_id=? AND profile_id=? ORDER BY created_at DESC LIMIT ?', (user_id, profile_id, max(1, min(int(limit or 30), 100)))).fetchall()
def clear_history(profile_id: int, user_id: int | None = None) -> int:
user_id = user_id or default_user_id()
with connect() as conn:
# Note: Manual automation log cleanup is scoped to the active profile and current user.
cur = conn.execute('DELETE FROM automation_history WHERE user_id=? AND profile_id=?', (user_id, profile_id))
return int(cur.rowcount or 0)
def _condition_true(t: dict[str, Any], cond: dict[str, Any]) -> bool: def _condition_true(t: dict[str, Any], cond: dict[str, Any]) -> bool:
typ = str(cond.get('type') or '') typ = str(cond.get('type') or '')
if typ == 'completed': return bool(int(t.get('complete') or 0)) if typ == 'completed': return bool(int(t.get('complete') or 0))

View File

@@ -30,6 +30,8 @@ def cleanup(force: bool = False) -> dict[str, int]:
targets = { targets = {
"traffic_history": ("created_at", TRAFFIC_HISTORY_RETENTION_DAYS), "traffic_history": ("created_at", TRAFFIC_HISTORY_RETENTION_DAYS),
"smart_queue_history": ("created_at", SMART_QUEUE_HISTORY_RETENTION_DAYS), "smart_queue_history": ("created_at", SMART_QUEUE_HISTORY_RETENTION_DAYS),
# Note: Automation history follows Smart Queue retention; rules and rule state are never deleted here.
"automation_history": ("created_at", SMART_QUEUE_HISTORY_RETENTION_DAYS),
"jobs": ("updated_at", JOBS_RETENTION_DAYS), "jobs": ("updated_at", JOBS_RETENTION_DAYS),
"logs": ("created_at", LOG_RETENTION_DAYS), "logs": ("created_at", LOG_RETENTION_DAYS),
} }

View File

@@ -132,6 +132,32 @@ def _excluded_hashes(profile_id: int, user_id: int) -> set[str]:
return {r['torrent_hash'] for r in list_exclusions(profile_id, user_id)} return {r['torrent_hash'] for r in list_exclusions(profile_id, user_id)}
def _label_names(value: str | None) -> list[str]:
names: list[str] = []
for part in str(value or '').replace(';', ',').replace('|', ',').split(','):
label = part.strip()
if label and label not in names:
names.append(label)
return names
def _label_value(labels: list[str]) -> str:
output: list[str] = []
for label in labels:
item = str(label or '').strip()
if item and item not in output:
output.append(item)
return ', '.join(output)
def _has_smart_queue_label(value: str | None) -> bool:
return SMART_QUEUE_LABEL in _label_names(value)
def _without_smart_queue_label(value: str | None) -> str:
return _label_value([label for label in _label_names(value) if label != SMART_QUEUE_LABEL])
def _remember_auto_label(profile_id: int, torrent_hash: str, previous_label: str) -> None: def _remember_auto_label(profile_id: int, torrent_hash: str, previous_label: str) -> None:
now = utcnow() now = utcnow()
with connect() as conn: with connect() as conn:
@@ -166,18 +192,18 @@ def _restore_auto_label(client: Any, profile_id: int, torrent_hash: str, current
).fetchone() ).fetchone()
live_label = _read_label(client, torrent_hash, current_label or '') live_label = _read_label(client, torrent_hash, current_label or '')
if not row: if not row:
if live_label != SMART_QUEUE_LABEL: if not _has_smart_queue_label(live_label):
return False return False
try: try:
# Note: Clear the Smart Queue label even when the torrent was marked earlier but no previous-label entry remains. # Note: Remove only the Smart Queue technical label and keep every user label untouched.
client.call('d.custom1.set', torrent_hash, '') client.call('d.custom1.set', torrent_hash, _without_smart_queue_label(live_label))
return True return True
except Exception: except Exception:
return False return False
previous = row.get('previous_label') or '' previous = row.get('previous_label') or ''
try: try:
# Note: On resume, Smart Queue restores the previous label only while it still sees its own technical label. # Note: Restore the saved label only when the current label still contains the Smart Queue marker.
if live_label == SMART_QUEUE_LABEL or current_label is None: if _has_smart_queue_label(live_label) or current_label is None:
client.call('d.custom1.set', torrent_hash, previous) client.call('d.custom1.set', torrent_hash, previous)
conn.execute('DELETE FROM smart_queue_auto_labels WHERE profile_id=? AND torrent_hash=?', (profile_id, torrent_hash)) conn.execute('DELETE FROM smart_queue_auto_labels WHERE profile_id=? AND torrent_hash=?', (profile_id, torrent_hash))
return True return True
@@ -282,10 +308,16 @@ def _read_live_start_state(client: Any, torrent_hash: str) -> dict[str, Any]:
result['started'] = bool(int(result.get('active') or 0)) result['started'] = bool(int(result.get('active') or 0))
return result return result
def _set_smart_queue_label(client: Any, torrent_hash: str, attempts: int = 3) -> bool: def _set_smart_queue_label(client: Any, torrent_hash: str, current_label: str = '', attempts: int = 3) -> bool:
labels = _label_names(current_label)
if SMART_QUEUE_LABEL in labels:
return True
labels.append(SMART_QUEUE_LABEL)
value = _label_value(labels)
for attempt in range(max(1, attempts)): for attempt in range(max(1, attempts)):
try: try:
client.call('d.custom1.set', torrent_hash, SMART_QUEUE_LABEL) # Note: Smart Queue appends its technical label instead of overwriting existing torrent labels.
client.call('d.custom1.set', torrent_hash, value)
return True return True
except Exception: except Exception:
if attempt < attempts - 1: if attempt < attempts - 1:
@@ -298,15 +330,15 @@ def _mark_auto_paused(client: Any, profile_id: int, torrent: dict[str, Any]) ->
if not torrent_hash: if not torrent_hash:
return False return False
previous = str(torrent.get('label') or '') previous = str(torrent.get('label') or '')
if previous != SMART_QUEUE_LABEL: if not _has_smart_queue_label(previous):
_remember_auto_label(profile_id, torrent_hash, previous) _remember_auto_label(profile_id, torrent_hash, previous)
return _set_smart_queue_label(client, torrent_hash) return _set_smart_queue_label(client, torrent_hash, previous)
def _is_smart_queue_hold(torrent: dict[str, Any] | None, manage_stopped: bool = True) -> bool: def _is_smart_queue_hold(torrent: dict[str, Any] | None, manage_stopped: bool = True) -> bool:
if not torrent or int(torrent.get('complete') or 0): if not torrent or int(torrent.get('complete') or 0):
return False return False
if str(torrent.get('label') or '') == SMART_QUEUE_LABEL: if _has_smart_queue_label(str(torrent.get('label') or '')):
return True return True
# Note: Paused in rTorrent usually has state=1 and active=0, so state=0 must not be required. # Note: Paused in rTorrent usually has state=1 and active=0, so state=0 must not be required.
# This lets Smart Queue treat paused torrents as pending and fill the queue target later. # This lets Smart Queue treat paused torrents as pending and fill the queue target later.
@@ -319,11 +351,11 @@ def _is_smart_queue_hold(torrent: dict[str, Any] | None, manage_stopped: bool =
def _clear_untracked_smart_queue_label(client: Any, torrent_hash: str, current_label: str) -> bool: def _clear_untracked_smart_queue_label(client: Any, torrent_hash: str, current_label: str) -> bool:
if current_label != SMART_QUEUE_LABEL: if not _has_smart_queue_label(current_label):
return False return False
try: try:
# Note: Clear an orphaned Smart Queue label when no previous-label entry exists in the database. # Note: Clear only the orphaned Smart Queue marker and keep unrelated labels intact.
client.call('d.custom1.set', torrent_hash, '') client.call('d.custom1.set', torrent_hash, _without_smart_queue_label(current_label))
return True return True
except Exception: except Exception:
return False return False
@@ -346,8 +378,8 @@ def _cleanup_auto_labels(client: Any, profile_id: int, torrents: list[dict[str,
if _restore_auto_label(client, profile_id, h, None if t is None else current_label): if _restore_auto_label(client, profile_id, h, None if t is None else current_label):
restored.append(h) restored.append(h)
continue continue
if current_label != SMART_QUEUE_LABEL: if not _has_smart_queue_label(current_label):
_set_smart_queue_label(client, h) _set_smart_queue_label(client, h, current_label)
for h, t in by_hash.items(): for h, t in by_hash.items():
if not h or h in keep_hashes or h in tracked_hashes or _is_smart_queue_hold(t, manage_stopped): if not h or h in keep_hashes or h in tracked_hashes or _is_smart_queue_hold(t, manage_stopped):
@@ -363,7 +395,7 @@ def _is_running_download_slot(t: dict[str, Any]) -> bool:
# Paused can have state=1/open=1, so a slot is counted only after d.is_active=1. # Paused can have state=1/open=1, so a slot is counted only after d.is_active=1.
if int(t.get('complete') or 0): if int(t.get('complete') or 0):
return False return False
if str(t.get('label') or '') == SMART_QUEUE_LABEL: if _has_smart_queue_label(str(t.get('label') or '')):
return False return False
status = str(t.get('status') or '').lower() status = str(t.get('status') or '').lower()
if status == 'checking' or status == 'paused' or bool(t.get('paused')): if status == 'checking' or status == 'paused' or bool(t.get('paused')):
@@ -375,7 +407,7 @@ def _is_waiting_download_candidate(t: dict[str, Any], manage_stopped: bool) -> b
"""Return True for paused/held torrents Smart Queue may resume later.""" """Return True for paused/held torrents Smart Queue may resume later."""
if int(t.get('complete') or 0): if int(t.get('complete') or 0):
return False return False
if str(t.get('label') or '') == SMART_QUEUE_LABEL: if _has_smart_queue_label(str(t.get('label') or '')):
return True return True
# Note: Paused items are the primary source for filling the queue, regardless of manage_stopped. # Note: Paused items are the primary source for filling the queue, regardless of manage_stopped.
if bool(t.get('paused')) or str(t.get('status') or '').lower() == 'paused': if bool(t.get('paused')) or str(t.get('status') or '').lower() == 'paused':
@@ -406,7 +438,7 @@ def check(profile: dict | None = None, user_id: int | None = None, force: bool =
excluded = _excluded_hashes(profile_id, user_id) excluded = _excluded_hashes(profile_id, user_id)
manage_stopped = bool(settings.get('manage_stopped')) manage_stopped = bool(settings.get('manage_stopped'))
def is_managed_hold(t: dict[str, Any]) -> bool: def is_managed_hold(t: dict[str, Any]) -> bool:
return str(t.get('label') or '') == SMART_QUEUE_LABEL return _has_smart_queue_label(str(t.get('label') or ''))
# Note: Count Smart Queue slots by d.is_active because Paused can have state=1/open=1 and must not occupy the limit. # Note: Count Smart Queue slots by d.is_active because Paused can have state=1/open=1 and must not occupy the limit.
downloading = [ downloading = [
@@ -538,7 +570,7 @@ def check(profile: dict | None = None, user_id: int | None = None, force: bool =
keep_labels = ( keep_labels = (
set(paused) set(paused)
| {str(t.get('hash') or '') for t in to_label_waiting} | {str(t.get('hash') or '') for t in to_label_waiting}
| {str(t.get('hash') or '') for t in stopped if str(t.get('label') or '') == SMART_QUEUE_LABEL and str(t.get('hash') or '') not in set(resumed)} | {str(t.get('hash') or '') for t in stopped if _has_smart_queue_label(str(t.get('label') or '')) and str(t.get('hash') or '') not in set(resumed)}
) )
restored = _cleanup_auto_labels(c, profile_id, torrents, keep_labels, manage_stopped) restored = _cleanup_auto_labels(c, profile_id, torrents, keep_labels, manage_stopped)
details = {'excluded': len(excluded), 'enabled': bool(settings.get('enabled')), 'auto_label': SMART_QUEUE_LABEL, 'labels_restored': restored, 'labels_failed': label_failed, 'start_failed': start_failed, 'start_no_effect': start_no_effect, 'start_results': start_results, 'resume_requested': resume_requested, 'active_verified': active_verified, 'waiting_labeled': len(to_label_waiting), 'manage_stopped': manage_stopped, 'max_active_downloads': max_active, 'active_before': len(downloading), 'active_after_expected': active_after_pause + len(resumed), 'paused_planned': len(to_pause), 'resumed_planned': len(to_resume), 'rtorrent_cap': rtorrent_cap} details = {'excluded': len(excluded), 'enabled': bool(settings.get('enabled')), 'auto_label': SMART_QUEUE_LABEL, 'labels_restored': restored, 'labels_failed': label_failed, 'start_failed': start_failed, 'start_no_effect': start_no_effect, 'start_results': start_results, 'resume_requested': resume_requested, 'active_verified': active_verified, 'waiting_labeled': len(to_label_waiting), 'manage_stopped': manage_stopped, 'max_active_downloads': max_active, 'active_before': len(downloading), 'active_after_expected': active_after_pause + len(resumed), 'paused_planned': len(to_pause), 'resumed_planned': len(to_resume), 'rtorrent_cap': rtorrent_cap}

File diff suppressed because one or more lines are too long

View File

@@ -1316,6 +1316,11 @@ body.mobile-mode .mobile-card {
font-size: 0.78rem; font-size: 0.78rem;
white-space: normal; white-space: normal;
} }
.automation-history-toolbar {
display: flex;
justify-content: flex-end;
margin-bottom: 0.5rem;
}
.automation-history-details { .automation-history-details {
max-width: min(620px, 60vw); max-width: min(620px, 60vw);
} }
@@ -1346,6 +1351,9 @@ body.mobile-mode .mobile-card {
grid-column: auto; grid-column: auto;
max-width: 100%; max-width: 100%;
} }
.automation-history-toolbar {
justify-content: flex-start;
}
} }
.disk-status { .disk-status {
display: inline-flex; display: inline-flex;