import pytest import os import requests import tempfile import zipfile from PyQt5 import QtCore, QtTest from .gui_base_test import GuiBaseTest class TestShare(GuiBaseTest): # Shared test methods def file_selection_widget_has_files(self, tab, num=3): """Test that the number of items in the list is as expected""" self.assertEqual( tab.get_mode().server_status.file_selection.get_num_files(), num ) def deleting_all_files_hides_delete_button(self, tab): """Test that clicking on the file item shows the delete button. Test that deleting the only item in the list hides the delete button""" rect = tab.get_mode().server_status.file_selection.file_list.visualItemRect( tab.get_mode().server_status.file_selection.file_list.item(0) ) QtTest.QTest.mouseClick( tab.get_mode().server_status.file_selection.file_list.viewport(), QtCore.Qt.LeftButton, pos=rect.center(), ) # Delete button should be visible self.assertTrue( tab.get_mode().server_status.file_selection.delete_button.isVisible() ) # Click delete, delete button should still be visible since we have one more file tab.get_mode().server_status.file_selection.delete_button.click() rect = tab.get_mode().server_status.file_selection.file_list.visualItemRect( tab.get_mode().server_status.file_selection.file_list.item(0) ) QtTest.QTest.mouseClick( tab.get_mode().server_status.file_selection.file_list.viewport(), QtCore.Qt.LeftButton, pos=rect.center(), ) self.assertTrue( tab.get_mode().server_status.file_selection.delete_button.isVisible() ) tab.get_mode().server_status.file_selection.delete_button.click() # No more files, the delete button should be hidden self.assertFalse( tab.get_mode().server_status.file_selection.delete_button.isVisible() ) def add_a_file_and_delete_using_its_delete_widget(self, tab): """Test that we can also delete a file by clicking on its [X] widget""" num_files = tab.get_mode().server_status.file_selection.get_num_files() tab.get_mode().server_status.file_selection.file_list.add_file(self.tmpfiles[0]) tab.get_mode().server_status.file_selection.file_list.item( 0 ).item_button.click() self.file_selection_widget_has_files(tab, num_files) def file_selection_widget_read_files(self, tab): """Re-add some files to the list so we can share""" num_files = tab.get_mode().server_status.file_selection.get_num_files() tab.get_mode().server_status.file_selection.file_list.add_file(self.tmpfiles[0]) tab.get_mode().server_status.file_selection.file_list.add_file(self.tmpfiles[1]) self.file_selection_widget_has_files(tab, num_files + 2) def add_delete_buttons_hidden(self, tab): """Test that the add and delete buttons are hidden when the server starts""" self.assertFalse( tab.get_mode().server_status.file_selection.add_button.isVisible() ) self.assertFalse( tab.get_mode().server_status.file_selection.delete_button.isVisible() ) def download_share(self, tab): """Test that we can download the share""" url = f"http://127.0.0.1:{tab.app.port}/download" if tab.settings.get("general", "public"): r = requests.get(url) else: r = requests.get( url, auth=requests.auth.HTTPBasicAuth( "onionshare", tab.get_mode().server_status.web.password ), ) tmp_file = tempfile.NamedTemporaryFile() with open(tmp_file.name, "wb") as f: f.write(r.content) zip = zipfile.ZipFile(tmp_file.name) QtTest.QTest.qWait(50) self.assertEqual("onionshare", zip.read("test.txt").decode("utf-8")) QtTest.QTest.qWait(500) def individual_file_is_viewable_or_not(self, tab): """ Test that an individual file is viewable (when in autostop_sharing is false) or that it isn't (when not in autostop_sharing is true) """ url = f"http://127.0.0.1:{tab.app.port}" download_file_url = f"http://127.0.0.1:{tab.app.port}/test.txt" if tab.settings.get("general", "public"): r = requests.get(url) else: r = requests.get( url, auth=requests.auth.HTTPBasicAuth( "onionshare", tab.get_mode().server_status.web.password ), ) if tab.settings.get("share", "autostop_sharing"): self.assertFalse('a href="/test.txt"' in r.text) if tab.settings.get("general", "public"): r = requests.get(download_file_url) else: r = requests.get( download_file_url, auth=requests.auth.HTTPBasicAuth( "onionshare", tab.get_mode().server_status.web.password ), ) self.assertEqual(r.status_code, 404) self.download_share(tab) else: self.assertTrue('a href="test.txt"' in r.text) if tab.settings.get("general", "public"): r = requests.get(download_file_url) else: r = requests.get( download_file_url, auth=requests.auth.HTTPBasicAuth( "onionshare", tab.get_mode().server_status.web.password ), ) tmp_file = tempfile.NamedTemporaryFile() with open(tmp_file.name, "wb") as f: f.write(r.content) with open(tmp_file.name, "r") as f: self.assertEqual("onionshare", f.read()) QtTest.QTest.qWait(500) def hit_401(self, tab): """Test that the server stops after too many 401s, or doesn't when in public_mode""" url = f"http://127.0.0.1:{tab.app.port}/" for _ in range(20): password_guess = self.gui.common.build_password() requests.get( url, auth=requests.auth.HTTPBasicAuth("onionshare", password_guess) ) # A nasty hack to avoid the Alert dialog that blocks the rest of the test if not tab.settings.get("general", "public"): QtCore.QTimer.singleShot(200, self.accept_dialog) # In public mode, we should still be running (no rate-limiting) if tab.settings.get("general", "public"): self.web_server_is_running(tab) # In non-public mode, we should be shut down (rate-limiting) else: self.web_server_is_stopped(tab) # Auto-start timer tests def set_autostart_timer(self, tab, timer): """Test that the timer can be set""" schedule = QtCore.QDateTime.currentDateTime().addSecs(timer) tab.get_mode().mode_settings_widget.autostart_timer_widget.setDateTime(schedule) self.assertTrue( tab.get_mode().mode_settings_widget.autostart_timer_widget.dateTime(), schedule, ) def autostart_timer_widget_hidden(self, tab): """Test that the auto-start timer widget is hidden when share has started""" self.assertFalse( tab.get_mode().mode_settings_widget.autostart_timer_widget.isVisible() ) def scheduled_service_started(self, tab, wait): """Test that the server has timed out after the timer ran out""" QtTest.QTest.qWait(wait) # We should have started now self.assertEqual(tab.get_mode().server_status.status, 2) def cancel_the_share(self, tab): """Test that we can cancel a share before it's started up """ self.server_working_on_start_button_pressed(tab) self.server_status_indicator_says_scheduled(tab) self.add_delete_buttons_hidden(tab) self.mode_settings_widget_is_hidden(tab) self.set_autostart_timer(tab, 10) QtTest.QTest.mousePress( tab.get_mode().server_status.server_button, QtCore.Qt.LeftButton ) QtTest.QTest.qWait(100) QtTest.QTest.mouseRelease( tab.get_mode().server_status.server_button, QtCore.Qt.LeftButton ) self.assertEqual(tab.get_mode().server_status.status, 0) self.server_is_stopped(tab) self.web_server_is_stopped(tab) # Grouped tests follow from here def run_all_share_mode_setup_tests(self, tab): """Tests in share mode prior to starting a share""" tab.get_mode().server_status.file_selection.file_list.add_file( self.tmpfile_test ) tab.get_mode().server_status.file_selection.file_list.add_file(self.tmpfiles[0]) tab.get_mode().server_status.file_selection.file_list.add_file(self.tmpfiles[1]) self.file_selection_widget_has_files(tab, 3) self.history_is_not_visible(tab) self.click_toggle_history(tab) self.history_is_visible(tab) self.deleting_all_files_hides_delete_button(tab) self.add_a_file_and_delete_using_its_delete_widget(tab) self.file_selection_widget_read_files(tab) def run_all_share_mode_started_tests(self, tab, startup_time=2000): """Tests in share mode after starting a share""" self.server_working_on_start_button_pressed(tab) self.server_status_indicator_says_starting(tab) self.add_delete_buttons_hidden(tab) self.mode_settings_widget_is_hidden(tab) self.server_is_started(tab, startup_time) self.web_server_is_running(tab) self.have_a_password(tab) self.url_description_shown(tab) self.have_copy_url_button(tab) self.server_status_indicator_says_started(tab) def run_all_share_mode_download_tests(self, tab): """Tests in share mode after downloading a share""" tab.get_mode().server_status.file_selection.file_list.add_file( self.tmpfile_test ) self.web_page(tab, "Total size") self.download_share(tab) self.history_widgets_present(tab) self.server_is_stopped(tab) self.web_server_is_stopped(tab) self.server_status_indicator_says_closed(tab) self.add_button_visible(tab) self.server_working_on_start_button_pressed(tab) self.toggle_indicator_is_reset(tab) self.server_is_started(tab) self.history_indicator(tab) def run_all_share_mode_individual_file_download_tests(self, tab): """Tests in share mode after downloading a share""" self.web_page(tab, "Total size") self.individual_file_is_viewable_or_not(tab) self.history_widgets_present(tab) self.server_is_stopped(tab) self.web_server_is_stopped(tab) self.server_status_indicator_says_closed(tab) self.add_button_visible(tab) self.server_working_on_start_button_pressed(tab) self.server_is_started(tab) self.history_indicator(tab) def run_all_share_mode_tests(self, tab): """End-to-end share tests""" self.run_all_share_mode_setup_tests(tab) self.run_all_share_mode_started_tests(tab) self.run_all_share_mode_download_tests(tab) def run_all_clear_all_button_tests(self, tab): """Test the Clear All history button""" self.run_all_share_mode_setup_tests(tab) self.run_all_share_mode_started_tests(tab) print( "history items: {}".format( len(tab.get_mode().history.item_list.items.keys()) ) ) self.individual_file_is_viewable_or_not(tab) print( "history items: {}".format( len(tab.get_mode().history.item_list.items.keys()) ) ) self.history_widgets_present(tab) self.clear_all_history_items(tab, 0) self.individual_file_is_viewable_or_not(tab) self.clear_all_history_items(tab, 2) def run_all_share_mode_individual_file_tests(self, tab): """Tests in share mode when viewing an individual file""" self.run_all_share_mode_setup_tests(tab) self.run_all_share_mode_started_tests(tab) self.run_all_share_mode_individual_file_download_tests(tab) # Tests @pytest.mark.gui def test_autostart_and_autostop_timer_mismatch(self): """ If autostart timer is after autostop timer, a warning should be thrown """ tab = self.new_share_tab() tab.get_mode().mode_settings_widget.toggle_advanced_button.click() tab.get_mode().mode_settings_widget.autostart_timer_checkbox.click() tab.get_mode().mode_settings_widget.autostop_timer_checkbox.click() def accept_dialog(): window = tab.common.gui.qtapp.activeWindow() if window: window.close() self.run_all_common_setup_tests() self.run_all_share_mode_setup_tests(tab) self.set_autostart_timer(tab, 15) self.set_timeout(tab, 5) QtCore.QTimer.singleShot(200, accept_dialog) tab.get_mode().server_status.server_button.click() self.server_is_stopped(tab) self.close_all_tabs() @pytest.mark.gui def test_autostart_timer(self): """ Autostart timer should automatically start """ tab = self.new_share_tab() tab.get_mode().mode_settings_widget.toggle_advanced_button.click() tab.get_mode().mode_settings_widget.autostart_timer_checkbox.click() self.run_all_common_setup_tests() self.run_all_share_mode_setup_tests(tab) self.set_autostart_timer(tab, 2) self.server_working_on_start_button_pressed(tab) self.autostart_timer_widget_hidden(tab) self.server_status_indicator_says_scheduled(tab) self.web_server_is_stopped(tab) self.scheduled_service_started(tab, 2200) self.web_server_is_running(tab) self.close_all_tabs() @pytest.mark.gui def test_autostart_timer_too_short(self): """ Autostart timer should throw a warning if the scheduled time is too soon """ tab = self.new_share_tab() tab.get_mode().mode_settings_widget.toggle_advanced_button.click() tab.get_mode().mode_settings_widget.autostart_timer_checkbox.click() def accept_dialog(): window = tab.common.gui.qtapp.activeWindow() if window: window.close() self.run_all_common_setup_tests() self.run_all_share_mode_setup_tests(tab) # Set a low timeout self.set_autostart_timer(tab, 2) QtTest.QTest.qWait(2200) QtCore.QTimer.singleShot(200, accept_dialog) tab.get_mode().server_status.server_button.click() self.assertEqual(tab.get_mode().server_status.status, 0) self.close_all_tabs() @pytest.mark.gui def test_autostart_timer_cancel(self): """ Test canceling a scheduled share """ tab = self.new_share_tab() tab.get_mode().mode_settings_widget.toggle_advanced_button.click() tab.get_mode().mode_settings_widget.autostart_timer_checkbox.click() self.run_all_common_setup_tests() self.run_all_share_mode_setup_tests(tab) self.cancel_the_share(tab) self.close_all_tabs() @pytest.mark.gui def test_clear_all_button(self): """ Test canceling a scheduled share """ tab = self.new_share_tab() tab.get_mode().autostop_sharing_checkbox.click() self.run_all_common_setup_tests() self.run_all_clear_all_button_tests(tab) self.close_all_tabs() @pytest.mark.gui def test_public_mode(self): """ Public mode shouldn't have a password """ tab = self.new_share_tab() tab.get_mode().mode_settings_widget.public_checkbox.click() self.run_all_common_setup_tests() self.run_all_share_mode_tests(tab) self.close_all_tabs() @pytest.mark.gui def test_without_autostop_sharing(self): """ Disable autostop sharing after first download """ tab = self.new_share_tab() tab.get_mode().autostop_sharing_checkbox.click() self.run_all_common_setup_tests() self.run_all_share_mode_tests(tab) self.close_all_tabs() @pytest.mark.gui def test_download(self): """ Test downloading in share mode """ tab = self.new_share_tab() self.run_all_common_setup_tests() self.run_all_share_mode_tests(tab) self.close_all_tabs() @pytest.mark.gui def test_individual_files_without_autostop_sharing(self): """ Test downloading individual files with autostop sharing disabled """ tab = self.new_share_tab() tab.get_mode().autostop_sharing_checkbox.click() self.run_all_common_setup_tests() self.run_all_share_mode_individual_file_tests(tab) self.close_all_tabs() @pytest.mark.gui def test_individual_files(self): """ Test downloading individual files """ tab = self.new_share_tab() self.run_all_common_setup_tests() self.run_all_share_mode_individual_file_tests(tab) self.close_all_tabs() @pytest.mark.gui def test_large_download(self): """ Test a large download """ tab = self.new_share_tab() self.run_all_common_setup_tests() self.run_all_share_mode_setup_tests(tab) tab.get_mode().server_status.file_selection.file_list.add_file( self.tmpfile_large ) self.run_all_share_mode_started_tests(tab, startup_time=15000) self.assertTrue(tab.get_mode().filesize_warning.isVisible()) self.download_share(tab) self.server_is_stopped(tab) self.web_server_is_stopped(tab) self.server_status_indicator_says_closed(tab) self.close_all_tabs() @pytest.mark.gui def test_persistent_password(self): """ Test a large download """ tab = self.new_share_tab() tab.get_mode().mode_settings_widget.persistent_checkbox.click() self.run_all_common_setup_tests() self.run_all_share_mode_setup_tests(tab) self.run_all_share_mode_started_tests(tab) password = tab.get_mode().server_status.web.password self.run_all_share_mode_download_tests(tab) self.run_all_share_mode_started_tests(tab) self.assertEqual(tab.get_mode().server_status.web.password, password) self.run_all_share_mode_download_tests(tab) self.close_all_tabs() @pytest.mark.gui def test_autostop_timer(self): """ Test the autostop timer """ tab = self.new_share_tab() tab.get_mode().mode_settings_widget.toggle_advanced_button.click() tab.get_mode().mode_settings_widget.autostop_timer_checkbox.click() self.run_all_common_setup_tests() self.run_all_share_mode_setup_tests(tab) self.set_timeout(tab, 5) self.run_all_share_mode_started_tests(tab) self.autostop_timer_widget_hidden(tab) self.server_timed_out(tab, 10000) self.web_server_is_stopped(tab) self.close_all_tabs() @pytest.mark.gui def test_autostop_timer_too_short(self): """ Test the autostop timer when the timeout is too short """ tab = self.new_share_tab() tab.get_mode().mode_settings_widget.toggle_advanced_button.click() tab.get_mode().mode_settings_widget.autostop_timer_checkbox.click() def accept_dialog(): window = tab.common.gui.qtapp.activeWindow() if window: window.close() self.run_all_common_setup_tests() self.run_all_share_mode_setup_tests(tab) # Set a low timeout self.set_timeout(tab, 2) QtTest.QTest.qWait(2100) QtCore.QTimer.singleShot(2200, accept_dialog) tab.get_mode().server_status.server_button.click() self.assertEqual(tab.get_mode().server_status.status, 0) self.close_all_tabs() @pytest.mark.gui def test_unreadable_file(self): """ Sharing an unreadable file should throw a warning """ tab = self.new_share_tab() def accept_dialog(): window = tab.common.gui.qtapp.activeWindow() if window: window.close() self.run_all_share_mode_setup_tests(tab) QtCore.QTimer.singleShot(200, accept_dialog) tab.get_mode().server_status.file_selection.file_list.add_file( "/tmp/nonexistent.txt" ) self.file_selection_widget_has_files(tab, 3) self.close_all_tabs()