forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			264 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			264 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* Any copyright is dedicated to the Public Domain.
 | |
|  * http://creativecommons.org/publicdomain/zero/1.0/ */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| requestLongerTimeout(2);
 | |
| 
 | |
| /**
 | |
|  * This test ensures that form data collection respects the privacy level as
 | |
|  * set by the user.
 | |
|  */
 | |
| add_task(function test_formdata() {
 | |
|   const URL = "http://mochi.test:8888/browser/browser/components/" +
 | |
|               "sessionstore/test/browser_formdata_sample.html";
 | |
| 
 | |
|   const OUTER_VALUE = "browser_formdata_" + Math.random();
 | |
|   const INNER_VALUE = "browser_formdata_" + Math.random();
 | |
| 
 | |
|   // Creates a tab, loads a page with some form fields,
 | |
|   // modifies their values and closes the tab.
 | |
|   function createAndRemoveTab() {
 | |
|     return Task.spawn(function () {
 | |
|       // Create a new tab.
 | |
|       let tab = gBrowser.addTab(URL);
 | |
|       let browser = tab.linkedBrowser;
 | |
|       yield promiseBrowserLoaded(browser);
 | |
| 
 | |
|       // Modify form data.
 | |
|       yield setInputValue(browser, {id: "txt", value: OUTER_VALUE});
 | |
|       yield setInputValue(browser, {id: "txt", value: INNER_VALUE, frame: 0});
 | |
| 
 | |
|       // Remove the tab.
 | |
|       yield promiseRemoveTab(tab);
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   yield createAndRemoveTab();
 | |
|   let [{state: {formdata}}] = JSON.parse(ss.getClosedTabData(window));
 | |
|   is(formdata.id.txt, OUTER_VALUE, "outer value is correct");
 | |
|   is(formdata.children[0].id.txt, INNER_VALUE, "inner value is correct");
 | |
| 
 | |
|   // Disable saving data for encrypted sites.
 | |
|   Services.prefs.setIntPref("browser.sessionstore.privacy_level", 1);
 | |
| 
 | |
|   yield createAndRemoveTab();
 | |
|   [{state: {formdata}}] = JSON.parse(ss.getClosedTabData(window));
 | |
|   is(formdata.id.txt, OUTER_VALUE, "outer value is correct");
 | |
|   ok(!formdata.children, "inner value was *not* stored");
 | |
| 
 | |
|   // Disable saving data for any site.
 | |
|   Services.prefs.setIntPref("browser.sessionstore.privacy_level", 2);
 | |
| 
 | |
|   yield createAndRemoveTab();
 | |
|   [{state: {formdata}}] = JSON.parse(ss.getClosedTabData(window));
 | |
|   ok(!formdata, "form data has *not* been stored");
 | |
| 
 | |
|   // Restore the default privacy level.
 | |
|   Services.prefs.clearUserPref("browser.sessionstore.privacy_level");
 | |
| });
 | |
| 
 | |
| /**
 | |
|  * This test ensures that a malicious website can't trick us into restoring
 | |
|  * form data into a wrong website and that we always check the stored URL
 | |
|  * before doing so.
 | |
|  */
 | |
| add_task(function test_url_check() {
 | |
|   const URL = "data:text/html;charset=utf-8,<input%20id=input>";
 | |
|   const VALUE = "value-" + Math.random();
 | |
| 
 | |
|   // Create a tab with an iframe containing an input field.
 | |
|   let tab = gBrowser.addTab(URL);
 | |
|   let browser = tab.linkedBrowser;
 | |
|   yield promiseBrowserLoaded(browser);
 | |
| 
 | |
|   // Restore a tab state with a given form data url.
 | |
|   function restoreStateWithURL(url) {
 | |
|     let state = {entries: [{url: URL}], formdata: {id: {input: VALUE}}};
 | |
| 
 | |
|     if (url) {
 | |
|       state.formdata.url = url;
 | |
|     }
 | |
| 
 | |
|     return promiseTabState(tab, state).then(() => getInputValue(browser, "input"));
 | |
|   }
 | |
| 
 | |
|   // Check that the form value is restored with the correct URL.
 | |
|   is((yield restoreStateWithURL(URL)), VALUE, "form data restored");
 | |
| 
 | |
|   // Check that the form value is *not* restored with the wrong URL.
 | |
|   is((yield restoreStateWithURL(URL + "?")), "", "form data not restored");
 | |
|   is((yield restoreStateWithURL()), "", "form data not restored");
 | |
| 
 | |
|   // Cleanup.
 | |
|   gBrowser.removeTab(tab);
 | |
| });
 | |
| 
 | |
| /**
 | |
|  * This test ensures that collecting form data works as expected when having
 | |
|  * nested frame sets.
 | |
|  */
 | |
| add_task(function test_nested() {
 | |
|   const URL = "data:text/html;charset=utf-8," +
 | |
|               "<iframe src='data:text/html;charset=utf-8," +
 | |
|               "<input autofocus=true>'/>";
 | |
| 
 | |
|   const FORM_DATA = {
 | |
|     children: [{
 | |
|       xpath: {"/xhtml:html/xhtml:body/xhtml:input": "M"},
 | |
|       url: "data:text/html;charset=utf-8,<input%20autofocus=true>"
 | |
|     }]
 | |
|   };
 | |
| 
 | |
|   // Create a tab with an iframe containing an input field.
 | |
|   let tab = gBrowser.selectedTab = gBrowser.addTab(URL);
 | |
|   let browser = tab.linkedBrowser;
 | |
|   yield promiseBrowserLoaded(browser);
 | |
| 
 | |
|   // Modify the input field's value.
 | |
|   yield sendMessage(browser, "ss-test:sendKeyEvent", {key: "m", frame: 0});
 | |
| 
 | |
|   // Remove the tab and check that we stored form data correctly.
 | |
|   yield promiseRemoveTab(tab);
 | |
|   let [{state: {formdata}}] = JSON.parse(ss.getClosedTabData(window));
 | |
|   is(JSON.stringify(formdata), JSON.stringify(FORM_DATA),
 | |
|     "formdata for iframe stored correctly");
 | |
| 
 | |
|   // Restore the closed tab.
 | |
|   tab = ss.undoCloseTab(window, 0);
 | |
|   browser = tab.linkedBrowser;
 | |
|   yield promiseTabRestored(tab);
 | |
| 
 | |
|   // Check that the input field has the right value.
 | |
|   TabState.flush(browser);
 | |
|   ({formdata} = JSON.parse(ss.getTabState(tab)));
 | |
|   is(JSON.stringify(formdata), JSON.stringify(FORM_DATA),
 | |
|     "formdata for iframe restored correctly");
 | |
| 
 | |
|   // Cleanup.
 | |
|   gBrowser.removeTab(tab);
 | |
| });
 | |
| 
 | |
| /**
 | |
|  * This test ensures that collecting form data for documents with
 | |
|  * designMode=on works as expected.
 | |
|  */
 | |
| add_task(function test_design_mode() {
 | |
|   const URL = "data:text/html;charset=utf-8,<h1>mozilla</h1>" +
 | |
|               "<script>document.designMode='on'</script>";
 | |
| 
 | |
|   // Load a tab with an editable document.
 | |
|   let tab = gBrowser.selectedTab = gBrowser.addTab(URL);
 | |
|   let browser = tab.linkedBrowser;
 | |
|   yield promiseBrowserLoaded(browser);
 | |
| 
 | |
|   // Modify the document content.
 | |
|   yield sendMessage(browser, "ss-test:sendKeyEvent", {key: "m"});
 | |
| 
 | |
|   // Close and restore the tab.
 | |
|   yield promiseRemoveTab(tab);
 | |
|   tab = ss.undoCloseTab(window, 0);
 | |
|   browser = tab.linkedBrowser;
 | |
|   yield promiseTabRestored(tab);
 | |
| 
 | |
|   // Check that the innerHTML value was restored.
 | |
|   let html = yield getInnerHTML(browser);
 | |
|   let expected = "<h1>Mmozilla</h1><script>document.designMode='on'</script>";
 | |
|   is(html, expected, "editable document has been restored correctly");
 | |
| 
 | |
|   // Close and restore the tab.
 | |
|   yield promiseRemoveTab(tab);
 | |
|   tab = ss.undoCloseTab(window, 0);
 | |
|   browser = tab.linkedBrowser;
 | |
|   yield promiseTabRestored(tab);
 | |
| 
 | |
|   // Check that the innerHTML value was restored.
 | |
|   html = yield getInnerHTML(browser);
 | |
|   expected = "<h1>Mmozilla</h1><script>document.designMode='on'</script>";
 | |
|   is(html, expected, "editable document has been restored correctly");
 | |
| 
 | |
|   // Cleanup.
 | |
|   gBrowser.removeTab(tab);
 | |
| });
 | |
| 
 | |
| /**
 | |
|  * This test ensures that credit card numbers in form data will not be
 | |
|  * collected, while numbers that don't look like credit card numbers will
 | |
|  * still be collected.
 | |
|  */
 | |
| add_task(function test_ccNumbers() {
 | |
|   const validCCNumbers = [
 | |
|     // 15 digits
 | |
|     "930771457288760", "474915027480942",
 | |
|     "924894781317325", "714816113937185",
 | |
|     "790466087343106", "474320195408363",
 | |
|     "219211148122351", "633038472250799",
 | |
|     "354236732906484", "095347810189325",
 | |
|     // 16 digits
 | |
|     "3091269135815020", "5471839082338112",
 | |
|     "0580828863575793", "5015290610002932",
 | |
|     "9465714503078607", "4302068493801686",
 | |
|     "2721398408985465", "6160334316984331",
 | |
|     "8643619970075142", "0218246069710785"
 | |
|   ];
 | |
| 
 | |
|   const invalidCCNumbers = [
 | |
|     // 15 digits
 | |
|     "526931005800649", "724952425140686",
 | |
|     "379761391174135", "030551436468583",
 | |
|     "947377014076746", "254848023655752",
 | |
|     "226871580283345", "708025346034339",
 | |
|     "917585839076788", "918632588027666",
 | |
|     // 16 digits
 | |
|     "9946177098017064", "4081194386488872",
 | |
|     "3095975979578034", "3662215692222536",
 | |
|     "6723210018630429", "4411962856225025",
 | |
|     "8276996369036686", "4449796938248871",
 | |
|     "3350852696538147", "5011802870046957"
 | |
|   ];
 | |
| 
 | |
|   const URL = "http://mochi.test:8888/browser/browser/components/" +
 | |
|               "sessionstore/test/browser_formdata_sample.html";
 | |
| 
 | |
|   // Creates a tab, loads a page with a form field, sets the value of the
 | |
|   // field, and then removes the tab to trigger data collection.
 | |
|   function* createAndRemoveTab(formValue) {
 | |
|     // Create a new tab.
 | |
|     let tab = gBrowser.addTab(URL);
 | |
|     let browser = tab.linkedBrowser;
 | |
|     yield promiseBrowserLoaded(browser);
 | |
| 
 | |
|     // Set form value.
 | |
|     yield setInputValue(browser, {id: "txt", value: formValue});
 | |
| 
 | |
|     // Remove the tab.
 | |
|     yield promiseRemoveTab(tab);
 | |
|   }
 | |
| 
 | |
|   // Test that valid CC numbers are not collected.
 | |
|   for (let number of validCCNumbers) {
 | |
|     yield createAndRemoveTab(number);
 | |
|     let [{state}] = JSON.parse(ss.getClosedTabData(window));
 | |
|     ok(!("formdata" in state), "valid CC numbers are not collected");
 | |
|   }
 | |
| 
 | |
|   // Test that non-CC numbers are still collected.
 | |
|   for (let number of invalidCCNumbers) {
 | |
|     yield createAndRemoveTab(number);
 | |
|     let [{state: {formdata}}] = JSON.parse(ss.getClosedTabData(window));
 | |
|     is(formdata.id.txt, number,
 | |
|        "numbers that are not valid CC numbers are still collected");
 | |
|   }
 | |
| });
 | |
| 
 | |
| function getInputValue(browser, id) {
 | |
|   return sendMessage(browser, "ss-test:getInputValue", {id: id});
 | |
| }
 | |
| 
 | |
| function setInputValue(browser, data) {
 | |
|   return sendMessage(browser, "ss-test:setInputValue", data);
 | |
| }
 | |
| 
 | |
| function getInnerHTML(browser) {
 | |
|   return sendMessage(browser, "ss-test:getInnerHTML", {selector: "body"});
 | |
| }
 | 
