Revision 587c1a88
Added by koszko over 1 year ago
| html/settings.html | ||
|---|---|---|
| 35 | 35 | #LOADCSS html/base.css | 
| 36 | 36 | #LOADCSS html/grid.css | 
| 37 | 37 | <style> | 
| 38 |       #loader, #indexeddb_error {
 | |
| 39 | margin: auto; | |
| 40 | padding: 1em; | |
| 41 | max-width: 800px; | |
| 42 | } | |
| 43 |       #indexeddb_error p {
 | |
| 44 | margin-bottom: 1em; | |
| 45 | } | |
| 46 |  | |
| 38 | 47 | /* Style top menu items. */ | 
| 39 | 48 |       #tab_heads>* {
 | 
| 40 | 49 | background-color: #70AF70; | 
| ... | ... | |
| 114 | 123 | #INCLUDE html/item_preview.html | 
| 115 | 124 | #INCLUDE html/text_entry_list.html | 
| 116 | 125 | #INCLUDE html/payload_create.html | 
| 117 | <ul id="tab_heads"> | |
| 118 | <li id="blocking_head"> Blocking </li> | |
| 119 | <li id="mappings_head"> Mappings </li> | |
| 120 | <li id="resources_head"> Resources </li> | |
| 121 | <li id="new_payload_head" class="active_head"> New payload </li> | |
| 122 | <li id="repos_head"> Repositories </li> | |
| 123 | </ul> | |
| 124 | <div id="top_menu_line" class="top_line"></div> | |
| 125 | <div id="blocking_tab" class="tab"> | |
| 126 | <div id="blocking_editable_container" class="grid_2"> | |
| 127 | <div id="blocking_list_container"> | |
| 128 | <h3>Block scripts on</h3> | |
| 129 | </div> | |
| 130 | <div id="allowing_list_container"> | |
| 131 | <h3>Allow scripts on</h3> | |
| 126 | <div id="loader"> | |
| 127 | Loading settings page... | |
| 128 | </div> | |
| 129 | <div id="indexeddb_error" class="hide"> | |
| 130 | <p> | |
| 131 | Cannot display settings page. | |
| 132 | </p> | |
| 133 | <p> | |
| 134 | Haketilo could not access IndexedDB. IndexedDB is an in-browser | |
| 135 | database in which Haketilo stores all its configuration. Without it, the | |
| 136 | settings page is non-operational. | |
| 137 | <p> | |
| 138 | #IF MOZILLA | |
| 139 | <div id="indexeddb_private_mode_explanation" class="hide"> | |
| 140 | <p> | |
| 141 | This issue is the result of using Haketilo in Private Browsing mode. | |
| 142 | For privacy reasons your browser blocks access to IndexedDB in | |
| 143 | private windows and this unfortunately also affects Haketilo's | |
| 144 | settings page. | |
| 145 | </p> | |
| 146 | <p> | |
| 147 | You can sacrifice this single privacy feature and enable IndexedDB | |
| 148 | access in private windows by navigating to "about:config" in the URL | |
| 149 | bar, agreeing to accept the risk and setting the | |
| 150 | "dom.indexedDB.privateBrowsing.enabled" preference to "true". Those | |
| 151 | pages that have their scripts blocked will still be unable to access | |
| 152 | IndexedDB. | |
| 153 | </p> | |
| 154 | <p> | |
| 155 | Alternatively, you can open Haketilo's settings in a separate, | |
| 156 | non-private window. The configuration you make there will take | |
| 157 | effect on pages opened in Private Browsing mode as well. | |
| 158 | </p> | |
| 132 | 159 | </div> | 
| 133 | <div id="default_policy_dialog" class="grid_col_both text_center"> | |
| 160 | #ENDIF | |
| 161 | </div> | |
| 162 | <div id="main_view" class="hide"> | |
| 163 | <ul id="tab_heads"> | |
| 164 | <li id="blocking_head"> Blocking </li> | |
| 165 | <li id="mappings_head"> Mappings </li> | |
| 166 | <li id="resources_head"> Resources </li> | |
| 167 | <li id="new_payload_head" class="active_head"> New payload </li> | |
| 168 | <li id="repos_head"> Repositories </li> | |
| 169 | </ul> | |
| 170 | <div id="top_menu_line" class="top_line"></div> | |
| 171 | <div id="blocking_tab" class="tab"> | |
| 172 | <div id="blocking_editable_container" class="grid_2"> | |
| 173 | <div id="blocking_list_container"> | |
| 174 | <h3>Block scripts on</h3> | |
| 175 | </div> | |
| 176 | <div id="allowing_list_container"> | |
| 177 | <h3>Allow scripts on</h3> | |
| 178 | </div> | |
| 179 | <div id="default_policy_dialog" class="grid_col_both text_center"> | |
| 134 | 180 | #INCLUDE html/default_blocking_policy.html | 
| 181 | </div> | |
| 135 | 182 | </div> | 
| 183 | <div id="blocking_dialog_container" class="hide"></div> | |
| 184 | </div> | |
| 185 | <div id="mappings_tab" class="tab"></div> | |
| 186 | <div id="resources_tab" class="tab"></div> | |
| 187 | <div id="new_payload_tab" class="tab active_tab"></div> | |
| 188 | <div id="repos_tab" class="tab"> | |
| 189 | <div id="repos_list_container"></div> | |
| 190 | <div id="repos_dialog_container" class="hide"></div> | |
| 136 | 191 | </div> | 
| 137 | <div id="blocking_dialog_container" class="hide"></div> | |
| 138 | </div> | |
| 139 | <div id="mappings_tab" class="tab"></div> | |
| 140 | <div id="resources_tab" class="tab"></div> | |
| 141 | <div id="new_payload_tab" class="tab active_tab"></div> | |
| 142 | <div id="repos_tab" class="tab"> | |
| 143 | <div id="repos_list_container"></div> | |
| 144 | <div id="repos_dialog_container" class="hide"></div> | |
| 145 | 192 | </div> | 
| 146 | 193 | #LOADJS html/settings.js | 
| 147 | 194 | </body> | 
| html/settings.js | ||
|---|---|---|
| 41 | 41 | * proprietary program, I am not going to enforce this in court. | 
| 42 | 42 | */ | 
| 43 | 43 |  | 
| 44 | #IMPORT common/indexeddb.js AS haketilodb | |
| 44 | 45 | #IMPORT html/dialog.js | 
| 45 | 46 |  | 
| 47 | #IF MOZILLA | |
| 48 | #FROM common/browser.js IMPORT browser | |
| 49 | #ENDIF | |
| 50 |  | |
| 46 | 51 | #FROM html/DOM_helpers.js IMPORT by_id | 
| 47 | 52 | #FROM html/text_entry_list.js IMPORT blocking_allowing_lists, repo_list | 
| 48 | 53 | #FROM html/item_list.js IMPORT mapping_list, resource_list | 
| ... | ... | |
| 118 | 123 | containers[0].append((await repo_list(dialog_ctx)).main_div); | 
| 119 | 124 | } | 
| 120 | 125 |  | 
| 121 | set_up_blocking_tab(); | |
| 122 | set_up_mappings_tab(); | |
| 123 | set_up_resources_tab(); | |
| 124 | set_up_new_payload_tab(); | |
| 125 | set_up_repos_tab(); | |
| 126 | function set_up_settings_view() {
 | |
| 127 |     by_id("loader").remove();
 | |
| 128 |     by_id("main_view").classList.remove("hide");
 | |
| 129 |  | |
| 130 | set_up_blocking_tab(); | |
| 131 | set_up_mappings_tab(); | |
| 132 | set_up_resources_tab(); | |
| 133 | set_up_new_payload_tab(); | |
| 134 | set_up_repos_tab(); | |
| 135 | } | |
| 136 |  | |
| 137 | #IF MOZILLA | |
| 138 | async function show_indexeddb_error() {
 | |
| 139 | const this_tab = await browser.tabs.getCurrent(); | |
| 140 | if (this_tab.incognito) | |
| 141 | 	by_id("indexeddb_private_mode_explanation").classList.remove("hide");
 | |
| 142 | #ELSE | |
| 143 | function show_indexeddb_error() {
 | |
| 144 | #ENDIF | |
| 145 |     by_id("loader").remove();
 | |
| 146 |     by_id("indexeddb_error").classList.remove("hide");
 | |
| 147 | } | |
| 148 |  | |
| 149 | async function init_settings_page() {
 | |
| 150 |     try {
 | |
| 151 | await haketilodb.get(); | |
| 152 |     } catch(e) {
 | |
| 153 | console.error(e); | |
| 154 | show_indexeddb_error(); | |
| 155 | return; | |
| 156 | } | |
| 157 |  | |
| 158 | set_up_settings_view(); | |
| 159 | } | |
| 160 |  | |
| 161 | #IF UNIT_TEST | |
| 162 | window.init_settings_page = init_settings_page; | |
| 163 | #ELSE | |
| 164 | init_settings_page(); | |
| 165 | #ENDIF | |
| test/haketilo_test/test_integration.py | ||
|---|---|---|
| 19 | 19 |  | 
| 20 | 20 | import pytest | 
| 21 | 21 |  | 
| 22 | from selenium.webdriver.support.ui import WebDriverWait | |
| 23 | from selenium.webdriver.support import expected_conditions as EC | |
| 24 | from selenium.webdriver.common.by import By | |
| 25 |  | |
| 22 | 26 | from .extension_crafting import get_extension_base_url | 
| 23 | 27 |  | 
| 24 | 28 | @pytest.mark.usefixtures('haketilo')
 | 
| ... | ... | |
| 30 | 34 | base_url = get_extension_base_url(driver) | 
| 31 | 35 | driver.get(base_url + 'html/settings.html') | 
| 32 | 36 |  | 
| 37 | WebDriverWait(driver, 10)\ | |
| 38 | .until(EC.visibility_of_element_located((By.ID, "main_view"))) | |
| 39 |  | |
| 33 | 40 | for tab_head_id, item_text in [ | 
| 34 | 41 |             ('resources_head', 'Haketilo demonstrational script'),
 | 
| 35 | 42 |             ('mappings_head',  'Haketilo demonstrational message'),
 | 
| test/haketilo_test/unit/test_indexeddb.py | ||
|---|---|---|
| 21 | 21 | import json | 
| 22 | 22 | from selenium.webdriver.common.by import By | 
| 23 | 23 | from selenium.webdriver.support.ui import WebDriverWait | 
| 24 | from selenium.webdriver.support import expected_conditions as EC | |
| 25 | 24 | from selenium.common.exceptions import WebDriverException | 
| 26 | 25 |  | 
| 27 | 26 | from ..script_loader import load_script | 
| test/haketilo_test/unit/test_settings.py | ||
|---|---|---|
| 20 | 20 | import pytest | 
| 21 | 21 | from .utils import * | 
| 22 | 22 |  | 
| 23 | from selenium.webdriver.support.ui import WebDriverWait | |
| 24 | from selenium.webdriver.support import expected_conditions as EC | |
| 25 | from selenium.webdriver.common.by import By | |
| 26 |  | |
| 23 | 27 | from ..extension_crafting import ExtraHTML | 
| 24 | 28 | from ..script_loader import load_script | 
| 25 | 29 | from .utils import * | 
| 26 | 30 |  | 
| 27 | @pytest.mark.ext_data({
 | |
| 31 | ext_data = {
 | |
| 28 | 32 | 'background_script': broker_js, | 
| 29 | 33 |     'extra_html': ExtraHTML('html/settings.html', wrap_into_htmldoc=False)
 | 
| 30 | }) | |
| 34 | } | |
| 35 |  | |
| 36 | @pytest.mark.ext_data(ext_data) | |
| 31 | 37 | @pytest.mark.usefixtures('webextension')
 | 
| 32 | 38 | def test_settings_page_tabs(driver, execute_in_page): | 
| 33 | 39 | """ | 
| 34 |     Test navigation throught the tabs of the settings page.
 | |
| 40 | Test navigation through the tabs of the settings page. | |
| 35 | 41 | """ | 
| 36 | 42 | # First, put some sample data in IndexedDB. | 
| 37 | 43 |     execute_in_page(load_script('common/indexeddb.js'))
 | 
| ... | ... | |
| 45 | 51 | # Now navigate to settings page. | 
| 46 | 52 |     testpage_url = driver.execute_script('return window.location.href;')
 | 
| 47 | 53 |     driver.get(testpage_url.replace('testpage.html', 'html/settings.html'))
 | 
| 54 |     execute_in_page('init_settings_page();')
 | |
| 55 |  | |
| 56 | WebDriverWait(driver, 10)\ | |
| 57 | .until(EC.visibility_of_element_located((By.ID, "main_view"))) | |
| 58 |  | |
| 59 |     assert driver.find_elements_by_id('loader') == []
 | |
| 60 |     assert not driver.find_element_by_id('indexeddb_error').is_displayed()
 | |
| 48 | 61 |  | 
| 49 | 62 | names = ['blocking', 'mappings', 'resources', 'new_payload', 'repos'] | 
| 50 | 63 |     tabs = dict([(n, driver.find_element_by_id(f'{n}_tab')) for n in names])
 | 
| ... | ... | |
| 61 | 74 |             assert 'active_head' not in heads[tab_name_2].get_attribute('class')
 | 
| 62 | 75 |             assert 'active_tab' not in tabs[tab_name_2].get_attribute('class')
 | 
| 63 | 76 | assert not tabs[tab_name_2].is_displayed() | 
| 77 |  | |
| 78 | @pytest.mark.ext_data({
 | |
| 79 | **ext_data, | |
| 80 | 'navigate_to': 'html/settings.html' | |
| 81 | }) | |
| 82 | @pytest.mark.usefixtures('webextension')
 | |
| 83 | @pytest.mark.parametrize('incognito', [True, False])
 | |
| 84 | def test_settings_page_indexeddb_error(driver, execute_in_page, incognito): | |
| 85 | """ | |
| 86 | Test if failure to access IndexedDB in settings page results in the | |
| 87 | appropriate message being shown. | |
| 88 | """ | |
| 89 | execute_in_page( | |
| 90 |         '''{
 | |
| 91 | /* | |
| 92 | * Mock an unavailable IndexedDB. Calling onerror() without having set | |
| 93 | * "errorCode" on the request is the behavior observed under Mozilla. | |
| 94 | */ | |
| 95 |         indexedDB.open = function() {
 | |
| 96 |             const dummy_open_request = {};
 | |
| 97 |             const dummy_event = {target: dummy_open_request};
 | |
| 98 | setTimeout(() => dummy_open_request.onerror(dummy_event), 0); | |
| 99 |  | |
| 100 | return dummy_open_request; | |
| 101 | } | |
| 102 |  | |
| 103 |         const dummy_tab = {incognito: arguments[0]};
 | |
| 104 | browser.tabs.getCurrent = () => Promise.resolve(dummy_tab); | |
| 105 |  | |
| 106 | init_settings_page(); | |
| 107 | }''', | |
| 108 | incognito) | |
| 109 |  | |
| 110 | WebDriverWait(driver, 10)\ | |
| 111 | .until(EC.visibility_of_element_located((By.ID, 'indexeddb_error'))) | |
| 112 |  | |
| 113 |     assert driver.find_elements_by_id('loader') == []
 | |
| 114 |     assert not driver.find_element_by_id('main_view').is_displayed()
 | |
| 115 |  | |
| 116 | if incognito: | |
| 117 |         assert driver.find_element_by_id('indexeddb_private_mode_explanation')\
 | |
| 118 | .is_displayed() | |
Also available in: Unified diff
display an informative message in settings page if IndexedDB cannot be accessed