mirror of
https://github.com/freedombox/FreedomBox.git
synced 2026-03-11 09:04:54 +00:00
- Since we are going to be an OpenID Provider, we need to fix the URLs that
other apps will be configured with for authentication. So change now from
/plinth to /freedombox. If done later, it will be harder since all the
configuration files for all dependent apps will need to be updated.
Tests:
- App availability checking works. Request goes to /freedombox URL
- Favicon is served properly and through /favicon.ico URL
- Redirection happens from / to /freedombox directly
- UI is available on /freedombox and on /plinth
- Manual page show /freedombox as the URL in two places
- Static files are successfully served from /freedombox URLs. URLs inside page
start with /freedombox
- backup, bepasty, calibre, config, dynamicdns, ejabberd, featherwiki, gitweb,
ikiwiki, kiwix, miniflux, names, openvpn, shadowsocks, shadowsocksserver,
sharing, shapshot, tiddlywiki, users, wireguard, jsxc, matrixsynapse, first
wizard, storage, samba, tags functional tests work. Backup/restore test for
matrixsynapse fails due to an unrelated bug (server not restarted after
restore).
- Setting the home page works:
- Having /plinth in the home page configuration works. Shows selection
correctly.
- Setting to app works. Shows selection correctly.
- Setting to user home page (sets /freedombox). Shows selection correctly.
- Setting to apache default works. Shows selection correctly.
- Changing back to FreedomBox service works. Shows selection correctly.
- Unit tests work
- Configuration page shows /freedombox in description but not /plinth
- Diagnostics show /freedombox in tests
- Roundcube URL link in email app has /freedombox
- email loads the page /.well-known/autoconfig/mail/config-v1.1.xml correctly
- email app shows /freedombox/apps/roundcube for /roundcube if roundcube is not
installed.
- networks: router configuration page shows URL starting with /freedombox.
- snapshot: Shows URL starting with /freedombox on the app page
- js licenses page uses /freedombox prefix for JSXC.
Signed-off-by: Sunil Mohan Adapa <sunil@medhas.org>
Reviewed-by: James Valleroy <jvalleroy@mailbox.org>
272 lines
9.9 KiB
Python
272 lines
9.9 KiB
Python
# SPDX-License-Identifier: AGPL-3.0-or-later
|
|
"""
|
|
Functional, browser based tests for backups app.
|
|
"""
|
|
|
|
import os
|
|
import tempfile
|
|
import time
|
|
import urllib.parse
|
|
|
|
import pytest
|
|
import requests
|
|
|
|
from plinth.tests import functional
|
|
|
|
pytestmark = [pytest.mark.system, pytest.mark.backups]
|
|
|
|
REMOTE_PATH = 'tester@localhost:~/backups'
|
|
|
|
|
|
@pytest.fixture(scope='module', autouse=True)
|
|
def fixture_background(session_browser):
|
|
"""Login and install the app."""
|
|
functional.login(session_browser)
|
|
functional.install(session_browser, 'bind')
|
|
functional.app_enable(session_browser, 'bind')
|
|
yield
|
|
functional.app_disable(session_browser, 'bind')
|
|
_backup_schedule_disable(session_browser)
|
|
|
|
|
|
@pytest.fixture(scope='session')
|
|
def downloaded_file_info():
|
|
return dict()
|
|
|
|
|
|
def test_browser_waits_after_restore(session_browser):
|
|
"""Test that browser waits for redirect after restoring a backup."""
|
|
functional.backup_create(session_browser, 'bind', 'test_backups')
|
|
functional.backup_restore(session_browser, 'bind', 'test_backups')
|
|
|
|
_open_main_page(session_browser)
|
|
time.sleep(5)
|
|
_assert_main_page_is_shown(session_browser)
|
|
|
|
|
|
def test_download_upload_restore(session_browser, downloaded_file_info):
|
|
"""Test download, upload, and restore a backup."""
|
|
functional.set_forwarders(session_browser, '1.1.1.1')
|
|
functional.backup_create(session_browser, 'bind', 'test_backups')
|
|
|
|
functional.set_forwarders(session_browser, '1.0.0.1')
|
|
_backup_download(session_browser, downloaded_file_info, 'test_backups')
|
|
_backup_restore_from_upload(session_browser, 'bind', downloaded_file_info)
|
|
|
|
assert functional.get_forwarders(session_browser) == '1.1.1.1'
|
|
|
|
|
|
def test_set_schedule(session_browser):
|
|
"""Test set a schedule for a repository."""
|
|
_backup_schedule_set(session_browser, enable=False, daily=1, weekly=2,
|
|
monthly=3, run_at=2, without_app='names')
|
|
|
|
_backup_schedule_set(session_browser, enable=True, daily=10, weekly=20,
|
|
monthly=30, run_at=15, without_app='firewall')
|
|
|
|
_backup_schedule_assert(session_browser, enable=True, daily=10, weekly=20,
|
|
monthly=30, run_at=15, without_app='firewall')
|
|
|
|
|
|
def test_remote_backup_location(session_browser):
|
|
"""Test remote backup location operations."""
|
|
_add_remote_backup_location(session_browser)
|
|
assert _has_remote_backup_location(session_browser)
|
|
|
|
_remove_remote_backup_location(session_browser)
|
|
assert not _has_remote_backup_location(session_browser)
|
|
|
|
# Add it again without providing SSH password.
|
|
_add_remote_backup_location(session_browser, False)
|
|
assert _has_remote_backup_location(session_browser)
|
|
|
|
_remove_remote_backup_location(session_browser)
|
|
assert not _has_remote_backup_location(session_browser)
|
|
|
|
|
|
def _assert_main_page_is_shown(session_browser):
|
|
assert (session_browser.url.endswith('/freedombox/')
|
|
or session_browser.url.endswith('/plinth/'))
|
|
|
|
|
|
def _backup_download(session_browser, downloaded_file_info, archive_name):
|
|
file_path = _download(session_browser, archive_name)
|
|
downloaded_file_info['path'] = file_path
|
|
|
|
|
|
def _backup_restore_from_upload(session_browser, app_name,
|
|
downloaded_file_info):
|
|
path = downloaded_file_info['path']
|
|
try:
|
|
_upload_and_restore(session_browser, app_name, path)
|
|
except Exception as err:
|
|
raise err
|
|
finally:
|
|
os.remove(path)
|
|
|
|
|
|
def _backup_schedule_assert(session_browser, enable, daily, weekly, monthly,
|
|
run_at, without_app):
|
|
schedule = _backup_schedule_get(session_browser)
|
|
assert schedule['enable'] == enable
|
|
assert schedule['daily'] == daily
|
|
assert schedule['weekly'] == weekly
|
|
assert schedule['monthly'] == monthly
|
|
assert schedule['run_at'] == run_at
|
|
assert len(schedule['without_apps']) == 1
|
|
assert schedule['without_apps'][0] == without_app
|
|
|
|
|
|
def _backup_schedule_disable(session_browser):
|
|
"""Disable schedule for the root repository."""
|
|
_backup_schedule_set(session_browser, False, 1, 2, 3, 2, 'names')
|
|
|
|
|
|
def _backup_schedule_get(browser):
|
|
"""Return the current schedule set for the root repository."""
|
|
functional.nav_to_module(browser, 'backups')
|
|
with functional.wait_for_page_update(browser):
|
|
browser.links.find_by_href(
|
|
'/freedombox/sys/backups/root/schedule/').first.click()
|
|
|
|
without_apps = []
|
|
elements = browser.find_by_name('backups_schedule-selected_apps')
|
|
for element in elements:
|
|
if not element.checked:
|
|
without_apps.append(element.value)
|
|
|
|
return {
|
|
'enable':
|
|
browser.find_by_name('backups_schedule-enabled').checked,
|
|
'daily':
|
|
int(browser.find_by_name('backups_schedule-daily_to_keep').value),
|
|
'weekly':
|
|
int(browser.find_by_name('backups_schedule-weekly_to_keep').value),
|
|
'monthly':
|
|
int(
|
|
browser.find_by_name('backups_schedule-monthly_to_keep').value
|
|
),
|
|
'run_at':
|
|
int(browser.find_by_name('backups_schedule-run_at_hour').value),
|
|
'without_apps':
|
|
without_apps
|
|
}
|
|
|
|
|
|
def _backup_schedule_set(browser, enable, daily, weekly, monthly, run_at,
|
|
without_app):
|
|
"""Set the schedule for root repository."""
|
|
functional.nav_to_module(browser, 'backups')
|
|
with functional.wait_for_page_update(browser):
|
|
browser.links.find_by_href(
|
|
'/freedombox/sys/backups/root/schedule/').first.click()
|
|
|
|
if enable:
|
|
browser.find_by_name('backups_schedule-enabled').check()
|
|
else:
|
|
browser.find_by_name('backups_schedule-enabled').uncheck()
|
|
|
|
browser.fill('backups_schedule-daily_to_keep', daily)
|
|
browser.fill('backups_schedule-weekly_to_keep', weekly)
|
|
browser.fill('backups_schedule-monthly_to_keep', monthly)
|
|
browser.fill('backups_schedule-run_at_hour', run_at)
|
|
functional.eventually(browser.find_by_css, args=['.select-all'])
|
|
browser.find_by_css('.select-all').first.check()
|
|
browser.find_by_css(f'input[value="{without_app}"]').first.uncheck()
|
|
functional.submit(browser, form_class='form-backups_schedule')
|
|
|
|
|
|
def _download_file_logged_in(browser, url, suffix=''):
|
|
"""Download a file from Plinth, pretend being logged in via cookies"""
|
|
if not url.startswith('http'):
|
|
current_url = urllib.parse.urlparse(browser.url)
|
|
url = '%s://%s%s' % (current_url.scheme, current_url.netloc, url)
|
|
cookies = browser.driver.get_cookies()
|
|
cookies = {cookie['name']: cookie['value'] for cookie in cookies}
|
|
response = requests.get(url, verify=False, cookies=cookies)
|
|
with tempfile.NamedTemporaryFile(delete=False, suffix=suffix) as temp_file:
|
|
for chunk in response.iter_content(chunk_size=128):
|
|
temp_file.write(chunk)
|
|
return temp_file.name
|
|
|
|
|
|
def _download(browser, archive_name=None):
|
|
"""Download a backup archive to a temporary file on disk."""
|
|
functional.nav_to_module(browser, 'backups')
|
|
href = f'/freedombox/sys/backups/root/download/{archive_name}/'
|
|
url = functional.base_url + href
|
|
file_path = _download_file_logged_in(browser, url, suffix='.tar.gz')
|
|
return file_path
|
|
|
|
|
|
def _open_main_page(browser):
|
|
"""Open the FreedomBox interface main page."""
|
|
with functional.wait_for_page_update(browser):
|
|
browser.links.find_by_href('/freedombox/').first.click()
|
|
|
|
|
|
def _upload_and_restore(browser, app_name, downloaded_file_path):
|
|
"""Upload a backup archive from the disk and perform restore operation."""
|
|
functional.nav_to_module(browser, 'backups')
|
|
with functional.wait_for_page_update(browser):
|
|
browser.links.find_by_href(
|
|
'/freedombox/sys/backups/upload/').first.click()
|
|
|
|
fileinput = browser.find_by_id('id_backups-file')
|
|
fileinput.fill(downloaded_file_path)
|
|
# submit upload form
|
|
functional.submit(browser, form_class='form-upload')
|
|
# submit restore form
|
|
with functional.wait_for_page_update(
|
|
browser, expected_url='/freedombox/sys/backups/'):
|
|
functional.submit(browser, form_class='form-restore')
|
|
|
|
|
|
def _has_remote_backup_location(browser) -> bool:
|
|
"""Return whether atleast one remote backup location is configured."""
|
|
functional.nav_to_module(browser, 'backups')
|
|
return browser.is_element_present_by_css(
|
|
f'.repository[data-repository-name="{REMOTE_PATH}"]')
|
|
|
|
|
|
def _add_remote_backup_location(browser, ssh_use_password=True):
|
|
"""Add a remote backup location."""
|
|
if _has_remote_backup_location(browser):
|
|
_remove_remote_backup_location(browser)
|
|
|
|
browser.links.find_by_href(
|
|
'/freedombox/sys/backups/repositories/add-remote/').first.click()
|
|
browser.find_by_name('repository').fill(REMOTE_PATH)
|
|
password = functional.get_password(
|
|
functional.config['DEFAULT']['username'])
|
|
if ssh_use_password:
|
|
browser.find_by_id('id_ssh_auth_type_1').check()
|
|
browser.find_by_name('ssh_password').fill(password)
|
|
else:
|
|
browser.find_by_id('id_ssh_auth_type_0').check()
|
|
|
|
browser.choose('id_encryption', 'repokey')
|
|
browser.find_by_name('encryption_passphrase').fill(password)
|
|
browser.find_by_name('confirm_encryption_passphrase').fill(password)
|
|
functional.submit(browser, form_class='form-add-remote-repository')
|
|
|
|
assert browser.is_text_present('Added new remote SSH repository.')
|
|
|
|
if 'ssh-verify' in browser.url:
|
|
_verify_host_key(browser)
|
|
|
|
|
|
def _remove_remote_backup_location(browser):
|
|
"""Remove the remote backup location with known remote path."""
|
|
repository = browser.find_by_css(
|
|
f'.repository[data-repository-name="{REMOTE_PATH}"]').first
|
|
repository.find_by_css('.repository-remove').first.click()
|
|
functional.submit(browser, form_class='form-remove-location')
|
|
|
|
|
|
def _verify_host_key(browser):
|
|
"""Verify the remote location's SSH host key."""
|
|
browser.find_by_name('ssh_public_key').first.click()
|
|
functional.submit(browser, form_class='form-verify-ssh-hostkey')
|
|
assert browser.is_text_present('SSH host verified.')
|