mirror of
https://github.com/dgtlmoon/changedetection.io.git
synced 2026-04-29 06:17:11 +00:00
116 lines
4.0 KiB
Python
116 lines
4.0 KiB
Python
#!/usr/bin/env python3
|
|
|
|
from .util import set_original_response, live_server_setup, wait_for_all_checks
|
|
from flask import url_for
|
|
import io
|
|
from zipfile import ZipFile
|
|
import re
|
|
import time
|
|
|
|
|
|
def test_backup(client, live_server, measure_memory_usage, datastore_path):
|
|
# live_server_setup(live_server) # Setup on conftest per function
|
|
|
|
set_original_response(datastore_path=datastore_path)
|
|
|
|
|
|
# Add our URL to the import page
|
|
res = client.post(
|
|
url_for("imports.import_page"),
|
|
data={"urls": url_for('test_endpoint', _external=True)+"?somechar=őőőőőőőő"},
|
|
follow_redirects=True
|
|
)
|
|
|
|
assert b"1 Imported" in res.data
|
|
wait_for_all_checks(client)
|
|
|
|
# Launch the thread in the background to create the backup
|
|
res = client.get(
|
|
url_for("backups.request_backup"),
|
|
follow_redirects=True
|
|
)
|
|
time.sleep(4)
|
|
|
|
res = client.get(
|
|
url_for("backups.index"),
|
|
follow_redirects=True
|
|
)
|
|
# Can see the download link to the backup
|
|
assert b'<a href="/backups/download/changedetection-backup-20' in res.data
|
|
assert b'Remove backups' in res.data
|
|
|
|
# Get the latest one
|
|
res = client.get(
|
|
url_for("backups.download_backup", filename="latest"),
|
|
follow_redirects=True
|
|
)
|
|
|
|
# Should get the right zip content type
|
|
assert res.content_type == "application/zip"
|
|
|
|
# Should be PK/ZIP stream
|
|
assert res.data.count(b'PK') >= 2
|
|
|
|
backup = ZipFile(io.BytesIO(res.data))
|
|
l = backup.namelist()
|
|
|
|
# Check for UUID-based txt files (history, snapshot, and last-checksum)
|
|
uuid4hex_txt = re.compile('^[a-f0-9]{8}-?[a-f0-9]{4}-?4[a-f0-9]{3}-?[89ab][a-f0-9]{3}-?[a-f0-9]{12}.*txt', re.I)
|
|
txt_files = list(filter(uuid4hex_txt.match, l))
|
|
# Should be three txt files in the archive (history, snapshot, and last-checksum)
|
|
assert len(txt_files) == 3
|
|
|
|
# Check for watch.json files (new format)
|
|
uuid4hex_json = re.compile('^[a-f0-9]{8}-?[a-f0-9]{4}-?4[a-f0-9]{3}-?[89ab][a-f0-9]{3}-?[a-f0-9]{12}/watch\.json$', re.I)
|
|
json_files = list(filter(uuid4hex_json.match, l))
|
|
# Should be one watch.json file in the archive (the imported watch)
|
|
assert len(json_files) == 1, f"Expected 1 watch.json file, found {len(json_files)}: {json_files}"
|
|
|
|
# Check for changedetection.json (settings file)
|
|
assert 'changedetection.json' in l, "changedetection.json should be in backup"
|
|
|
|
# Get the latest one
|
|
res = client.get(
|
|
url_for("backups.remove_backups"),
|
|
follow_redirects=True
|
|
)
|
|
|
|
assert b'No backups found.' in res.data
|
|
|
|
|
|
def test_watch_data_package_download(client, live_server, measure_memory_usage, datastore_path):
|
|
"""Test downloading a single watch's data as a zip package"""
|
|
import os
|
|
|
|
set_original_response(datastore_path=datastore_path)
|
|
|
|
uuid = client.application.config.get('DATASTORE').add_watch(url=url_for('test_endpoint', _external=True))
|
|
client.get(url_for("ui.form_watch_checknow"), follow_redirects=True)
|
|
|
|
wait_for_all_checks(client)
|
|
|
|
# Download the watch data package
|
|
res = client.get(url_for("ui.ui_edit.watch_get_data_package", uuid=uuid))
|
|
|
|
# Should get the right zip content type
|
|
assert res.content_type == "application/zip"
|
|
|
|
# Should be PK/ZIP stream (PKzip header)
|
|
assert res.data[:2] == b'PK', "File should start with PK (PKzip header)"
|
|
assert res.data.count(b'PK') >= 2, "Should have multiple PK markers (zip file structure)"
|
|
|
|
# Verify zip contents
|
|
backup = ZipFile(io.BytesIO(res.data))
|
|
files = backup.namelist()
|
|
|
|
# Should have files in a UUID directory
|
|
assert any(uuid in f for f in files), f"Files should be in UUID directory: {files}"
|
|
|
|
# Should contain watch.json
|
|
watch_json_path = f"{uuid}/watch.json"
|
|
assert watch_json_path in files, f"Should contain watch.json, got: {files}"
|
|
|
|
# Should contain history/snapshot files
|
|
uuid4hex_txt = re.compile(f'^{re.escape(uuid)}/.*\\.txt', re.I)
|
|
txt_files = list(filter(uuid4hex_txt.match, files))
|
|
assert len(txt_files) > 0, f"Should have at least one .txt file (history/snapshot), got: {files}" |