Bug 968727 - Move common test runner parts in external helper files. r=bgrins, a=test-only
authorPatrick Brosset <pbrosset@mozilla.com>
Wed, 26 Mar 2014 14:38:06 +0100
changeset 192932 a1a02bbf28e9373b80d196576281199b1f2b15d4
parent 192931 4c267ebca47de79eb5d237c9b44d9c9588dec70a
child 192933 d429abe1d569266e5fc8737edce7d52b28364d98
push id474
push userasasaki@mozilla.com
push dateMon, 02 Jun 2014 21:01:02 +0000
treeherdermozilla-release@967f4cf1b31c [default view] [failures only]
perfherder[talos] [build metrics] [platform microbench] (compared to previous push)
reviewersbgrins, test-only
bugs968727
milestone30.0a2
Bug 968727 - Move common test runner parts in external helper files. r=bgrins, a=test-only
browser/devtools/markupview/test/browser.ini
browser/devtools/markupview/test/browser_markupview_copy_image_data.js
browser/devtools/markupview/test/browser_markupview_css_completion_style_attribute.js
browser/devtools/markupview/test/browser_markupview_highlight_hover_01.js
browser/devtools/markupview/test/browser_markupview_highlight_hover_02.js
browser/devtools/markupview/test/browser_markupview_html_edit_01.js
browser/devtools/markupview/test/browser_markupview_html_edit_02.js
browser/devtools/markupview/test/browser_markupview_html_edit_03.js
browser/devtools/markupview/test/browser_markupview_image_tooltip.js
browser/devtools/markupview/test/browser_markupview_mutation_01.js
browser/devtools/markupview/test/browser_markupview_mutation_02.js
browser/devtools/markupview/test/browser_markupview_navigation.js
browser/devtools/markupview/test/browser_markupview_pagesize_01.js
browser/devtools/markupview/test/browser_markupview_pagesize_02.js
browser/devtools/markupview/test/browser_markupview_search_01.js
browser/devtools/markupview/test/browser_markupview_tag_edit_01.js
browser/devtools/markupview/test/browser_markupview_tag_edit_05.js
browser/devtools/markupview/test/browser_markupview_tag_edit_06.js
browser/devtools/markupview/test/browser_markupview_tag_edit_07.js
browser/devtools/markupview/test/head.js
browser/devtools/markupview/test/helper_attributes_test_runner.js
browser/devtools/markupview/test/helper_outerhtml_test_runner.js
--- a/browser/devtools/markupview/test/browser.ini
+++ b/browser/devtools/markupview/test/browser.ini
@@ -5,16 +5,18 @@ support-files =
   doc_markup_flashing.html
   doc_markup_mutation.html
   doc_markup_navigation.html
   doc_markup_pagesize_01.html
   doc_markup_pagesize_02.html
   doc_markup_search.html
   doc_markup_tooltip.png
   head.js
+  helper_attributes_test_runner.js
+  helper_outerhtml_test_runner.js
 
 [browser_markupview_copy_image_data.js]
 [browser_markupview_css_completion_style_attribute.js]
 [browser_markupview_highlight_hover_01.js]
 [browser_markupview_highlight_hover_02.js]
 [browser_markupview_html_edit_01.js]
 [browser_markupview_html_edit_02.js]
 [browser_markupview_html_edit_03.js]
--- a/browser/devtools/markupview/test/browser_markupview_copy_image_data.js
+++ b/browser/devtools/markupview/test/browser_markupview_copy_image_data.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test that image nodes have the "copy data-uri" contextual menu item enabled
 // and that clicking it puts the image data into the clipboard
 
 const PAGE_CONTENT = [
   '<div></div>',
   '<img class="data" src="" />',
   '<canvas class="canvas" width="600" height="600"></canvas>'
 ].join("\n");
--- a/browser/devtools/markupview/test/browser_markupview_css_completion_style_attribute.js
+++ b/browser/devtools/markupview/test/browser_markupview_css_completion_style_attribute.js
@@ -1,11 +1,13 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
-   http://creativecommons.org/publicdomain/zero/1.0/ */
+ http://creativecommons.org/publicdomain/zero/1.0/ */
+
+"use strict";
 
 // Test CSS state is correctly determined and the corresponding suggestions are
 // displayed. i.e. CSS property suggestions are shown when cursor is like:
 // ```style="di|"``` where | is the cursor; And CSS value suggestion is
 // displayed when the cursor is like: ```style="display:n|"``` properly. No
 // suggestions should ever appear when the attribute is not a style attribute.
 // The correctness and cycling of the suggestions is covered in the ruleview
 // tests.
--- a/browser/devtools/markupview/test/browser_markupview_highlight_hover_01.js
+++ b/browser/devtools/markupview/test/browser_markupview_highlight_hover_01.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test that when first hovering over a node and immediately after selecting it
 // by clicking on it leaves the highlighter visible for as long as the mouse is
 // over the node
 
 let test = asyncTest(function*() {
   let {inspector} = yield addTab("data:text/html,<p>It's going to be legen....</p>").then(openInspector);
   let p = getNode("p");
 
--- a/browser/devtools/markupview/test/browser_markupview_highlight_hover_02.js
+++ b/browser/devtools/markupview/test/browser_markupview_highlight_hover_02.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test that when after an element is selected and highlighted on hover, if the
 // mouse leaves the markup-view and comes back again on the same element, that
 // the highlighter is shown again on the node
 
 let test = asyncTest(function*() {
   let {inspector} = yield addTab("data:text/html,<p>Select me!</p>").then(openInspector);
 
   info("hover over the <p> line in the markup-view so that it's the currently hovered node");
--- a/browser/devtools/markupview/test/browser_markupview_html_edit_01.js
+++ b/browser/devtools/markupview/test/browser_markupview_html_edit_01.js
@@ -1,14 +1,18 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test outerHTML edition via the markup-view
 
+loadHelperScript("helper_outerhtml_test_runner.js");
+
 const TEST_DATA = [
   {
     selector: "#one",
     oldHTML: '<div id="one">First <em>Div</em></div>',
     newHTML: '<div id="one">First Div</div>',
     validate: function(pageNode, selectedNode) {
       is(pageNode.textContent, "First Div", "New div has expected text content");
       ok(!getNode("#one em"), "No em remaining")
--- a/browser/devtools/markupview/test/browser_markupview_html_edit_02.js
+++ b/browser/devtools/markupview/test/browser_markupview_html_edit_02.js
@@ -1,14 +1,18 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test outerHTML edition via the markup-view
 
+loadHelperScript("helper_outerhtml_test_runner.js");
+
 const TEST_DATA = [
   {
     selector: "#badMarkup1",
     oldHTML: '<div id="badMarkup1">badMarkup1</div>',
     newHTML: '<div id="badMarkup1">badMarkup1</div> hanging</div>',
     validate: function(pageNode, selectedNode) {
       is(pageNode, selectedNode, "Original element is selected");
 
--- a/browser/devtools/markupview/test/browser_markupview_html_edit_03.js
+++ b/browser/devtools/markupview/test/browser_markupview_html_edit_03.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test that outerHTML editing keybindings work as expected and that *special*
 // elements like <html>, <body> and <head> can be edited correctly.
 
 const TEST_URL = "data:text/html," +
   "<!DOCTYPE html>" +
   "<head><meta charset='utf-8' /></head>" +
   "<body>" +
   "<div id=\"keyboard\"></div>" +
--- a/browser/devtools/markupview/test/browser_markupview_image_tooltip.js
+++ b/browser/devtools/markupview/test/browser_markupview_image_tooltip.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test that image preview tooltips are shown on img and canvas tags in the
 // markup-view and that the tooltip actually contains an image and shows the
 // right dimension label
 
 const PAGE_CONTENT = [
   '<img class="local" src="chrome://branding/content/about-logo.png" />',
   '<img class="data" src="" />',
   '<img class="remote" src="' + TEST_URL_ROOT + 'doc_markup_tooltip.png" />',
--- a/browser/devtools/markupview/test/browser_markupview_mutation_01.js
+++ b/browser/devtools/markupview/test/browser_markupview_mutation_01.js
@@ -1,20 +1,20 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
-/**
- * Tests that various mutations to the dom update the markup view correctly.
- * The test for comparing the markup view to the real dom is a bit weird:
- * - Select the text in the markup view
- * - Parse that as innerHTML in a document we've created for the purpose.
- * - Remove extraneous whitespace in that tree
- * - Compare it to the real dom with isEqualNode.
- */
+"use strict";
+
+// Tests that various mutations to the dom update the markup view correctly.
+// The test for comparing the markup view to the real dom is a bit weird:
+// - Select the text in the markup view
+// - Parse that as innerHTML in a document we've created for the purpose.
+// - Remove extraneous whitespace in that tree
+// - Compare it to the real dom with isEqualNode.
 
 const TEST_URL = TEST_URL_ROOT + "doc_markup_mutation.html";
 // All the mutation types we want to test.
 const TEST_DATA = [
   {
     desc: "Adding an attribute",
     test: () => {
       let node1 = getNode("#node1");
--- a/browser/devtools/markupview/test/browser_markupview_mutation_02.js
+++ b/browser/devtools/markupview/test/browser_markupview_mutation_02.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test that markup-containers in the markup-view do flash when their
 // corresponding DOM nodes mutate
 
 const TEST_URL = TEST_URL_ROOT + "doc_markup_flashing.html";
 // The test data contains a list of mutations to test.
 // Each item is an object:
 // - desc: a description of the test step, for better logging
 // - mutate: a function that should make changes to the content DOM
--- a/browser/devtools/markupview/test/browser_markupview_navigation.js
+++ b/browser/devtools/markupview/test/browser_markupview_navigation.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test that the markup-view nodes can be navigated to with the keyboard
 
 const TEST_URL = TEST_URL_ROOT + "doc_markup_navigation.html";
 const TEST_DATA = [
   ["pageup", "*doctype*"],
   ["down", "html"],
   ["down", "head"],
   ["down", "body"],
--- a/browser/devtools/markupview/test/browser_markupview_pagesize_01.js
+++ b/browser/devtools/markupview/test/browser_markupview_pagesize_01.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Tests that the markup view loads only as many nodes as specified by the
 // devtools.markup.pagesize preference.
 
 Services.prefs.setIntPref("devtools.markup.pagesize", 5);
 
 const TEST_URL = TEST_URL_ROOT + "doc_markup_pagesize_01.html";
 const TEST_DATA = [{
   desc: "Select the last item",
--- a/browser/devtools/markupview/test/browser_markupview_pagesize_02.js
+++ b/browser/devtools/markupview/test/browser_markupview_pagesize_02.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Tests that the markup view loads only as many nodes as specified
 // by the devtools.markup.pagesize preference and that pressing the "show all nodes"
 // actually shows the nodes
 
 const TEST_URL = TEST_URL_ROOT + "doc_markup_pagesize_02.html";
 
 // Make sure nodes are hidden when there are more than 5 in a row
 Services.prefs.setIntPref("devtools.markup.pagesize", 5);
--- a/browser/devtools/markupview/test/browser_markupview_search_01.js
+++ b/browser/devtools/markupview/test/browser_markupview_search_01.js
@@ -1,12 +1,14 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
+"use strict";
+
 // Test that searching for nodes using the selector-search input expands and
 // selects the right nodes in the markup-view, even when those nodes are deeply
 // nested (and therefore not attached yet when the markup-view is initialized).
 
 const TEST_URL = TEST_URL_ROOT + "doc_markup_search.html";
 
 let test = asyncTest(function*() {
   let {inspector, toolbox} = yield addTab(TEST_URL).then(openInspector);
--- a/browser/devtools/markupview/test/browser_markupview_tag_edit_01.js
+++ b/browser/devtools/markupview/test/browser_markupview_tag_edit_01.js
@@ -1,16 +1,18 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
 "use strict";
 
 // Test editing various markup-containers' attribute fields
 
+loadHelperScript("helper_attributes_test_runner.js");
+
 const TEST_URL = TEST_URL_ROOT + "doc_markup_edit.html";
 let TEST_DATA = [{
   desc: "Change an attribute",
   node: "#node1",
   originalAttributes: {
     id: "node1",
     class: "node1"
   },
--- a/browser/devtools/markupview/test/browser_markupview_tag_edit_05.js
+++ b/browser/devtools/markupview/test/browser_markupview_tag_edit_05.js
@@ -1,22 +1,23 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
 "use strict";
 
-/**
- * Tests that adding various types of attributes to nodes in the markup-view
- * works as expected. Also checks that the changes are properly undoable and
- * redoable. For each step in the test, we:
- * - Create a new DIV
- * - Make the change, check that the change was made as we expect
- * - Undo the change, check that the node is back in its original state
- * - Redo the change, check that the node change was made again correctly.
- */
+// Tests that adding various types of attributes to nodes in the markup-view
+// works as expected. Also checks that the changes are properly undoable and
+// redoable. For each step in the test, we:
+// - Create a new DIV
+// - Make the change, check that the change was made as we expect
+// - Undo the change, check that the node is back in its original state
+// - Redo the change, check that the node change was made again correctly.
+
+loadHelperScript("helper_attributes_test_runner.js");
 
 let TEST_URL = "data:text/html,<div>markup-view attributes addition test</div>";
 let TEST_DATA = [{
   desc: "Add an attribute value without closing \"",
   text: 'style="display: block;',
   expectedAttributes: {
     style: "display: block;"
   }
--- a/browser/devtools/markupview/test/browser_markupview_tag_edit_06.js
+++ b/browser/devtools/markupview/test/browser_markupview_tag_edit_06.js
@@ -1,22 +1,23 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
 "use strict";
 
-/**
- * Tests that adding various types of attributes to nodes in the markup-view
- * works as expected. Also checks that the changes are properly undoable and
- * redoable. For each step in the test, we:
- * - Create a new DIV
- * - Make the change, check that the change was made as we expect
- * - Undo the change, check that the node is back in its original state
- * - Redo the change, check that the node change was made again correctly.
- */
+// Tests that adding various types of attributes to nodes in the markup-view
+// works as expected. Also checks that the changes are properly undoable and
+// redoable. For each step in the test, we:
+// - Create a new DIV
+// - Make the change, check that the change was made as we expect
+// - Undo the change, check that the node is back in its original state
+// - Redo the change, check that the node change was made again correctly.
+
+loadHelperScript("helper_attributes_test_runner.js");
 
 let TEST_URL = "data:text/html,<div>markup-view attributes addition test</div>";
 let TEST_DATA = [{
   desc: "Mixed single and double quotes",
   text: "name=\"hi\" maxlength='not a number'",
   expectedAttributes: {
     maxlength: "not a number",
     name: "hi"
--- a/browser/devtools/markupview/test/browser_markupview_tag_edit_07.js
+++ b/browser/devtools/markupview/test/browser_markupview_tag_edit_07.js
@@ -1,16 +1,19 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
 "use strict";
 
 // One more test testing various add-attributes configurations
 // Some of the test data below asserts that long attributes get collapsed
 
+loadHelperScript("helper_attributes_test_runner.js");
+
 const LONG_ATTRIBUTE = "ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ";
 const LONG_ATTRIBUTE_COLLAPSED = "ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEF\u2026UVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ";
 const DATA_URL_INLINE_STYLE='color: red; background: url("");';
 const DATA_URL_INLINE_STYLE_COLLAPSED='color: red; background: url("\u2026NDDeNGe4Ug9C9zwz3gVLMDA/A6P9/AFGGFyjOXZtQAAAAAElFTkSuQmCC");';
 const DATA_URL_ATTRIBUTE = "";
 const DATA_URL_ATTRIBUTE_COLLAPSED = "\u20269/AFGGFyjOXZtQAAAAAElFTkSuQmCC";
 
 let TEST_URL = "data:text/html,<div>markup-view attributes addition test</div>";
--- a/browser/devtools/markupview/test/head.js
+++ b/browser/devtools/markupview/test/head.js
@@ -67,16 +67,32 @@ function addTab(url) {
     }, content);
   }, true);
   content.location = url;
 
   return def.promise;
 }
 
 /**
+ * Some tests may need to import one or more of the test helper scripts.
+ * A test helper script is simply a js file that contains common test code that
+ * is either not common-enough to be in head.js, or that is located in a separate
+ * directory.
+ * The script will be loaded synchronously and in the test's scope.
+ * @param {String} filePath The file path, relative to the current directory.
+ *                 Examples:
+ *                 - "helper_attributes_test_runner.js"
+ *                 - "../../../commandline/test/helpers.js"
+ */
+function loadHelperScript(filePath) {
+  let testDir = gTestPath.substr(0, gTestPath.lastIndexOf("/"));
+  Services.scriptloader.loadSubScript(testDir + "/" + filePath, this);
+}
+
+/**
  * Reload the current page
  * @return a promise that resolves when the inspector has emitted the event
  * new-root
  */
 function reloadPage(inspector) {
   info("Reloading the page");
   let newRoot = inspector.once("new-root");
   content.location.reload();
@@ -331,234 +347,8 @@ function getSelectorSearchBox(inspector)
  */
 function searchUsingSelectorSearch(selector, inspector) {
   info("Entering \"" + selector + "\" into the selector-search input field");
   let field = getSelectorSearchBox(inspector);
   field.focus();
   field.value = selector;
   EventUtils.sendKey("return", inspector.panelWin);
 }
-
-/**
- * Run a series of add-attributes tests.
- * This function will iterate over the provided tests array and run each test.
- * Each test's goal is to provide some text to be entered into the test node's
- * new-attribute field and check that the given attributes have been created.
- * After each test has run, the markup-view's undo command will be called and
- * the test runner will check if all the new attributes are gone.
- * @param {Array} tests See runAddAttributesTest for the structure
- * @param {DOMNode|String} nodeOrSelector The node or node selector
- * corresponding to an element on the current test page that has *no attributes*
- * when the test starts. It will be used to add and remove attributes.
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently
- * opened
- * @return a promise that resolves when the tests have run
- */
-function runAddAttributesTests(tests, nodeOrSelector, inspector) {
-  info("Running " + tests.length + " add-attributes tests");
-  return Task.spawn(function*() {
-    info("Selecting the test node");
-    let div = getNode("div");
-    yield selectNode(div, inspector);
-
-    for (let test of tests) {
-      yield runAddAttributesTest(test, div, inspector);
-    }
-
-    yield inspector.once("inspector-updated");
-  });
-}
-
-/**
- * Run a single add-attribute test.
- * See runAddAttributesTests for a description.
- * @param {Object} test A test object should contain the following properties:
- *        - desc {String} a textual description for that test, to help when
- *        reading logs
- *        - text {String} the string to be inserted into the new attribute field
- *        - expectedAttributes {Object} a key/value pair object that will be
- *        used to check the attributes on the test element
- *        - validate {Function} optional extra function that will be called after
- *        the attributes have been added and which should be used to assert some
- *        more things this test runner might not be checking. The function will
- *        be called with the following arguments:
- *          - {DOMNode} The element being tested
- *          - {MarkupContainer} The corresponding container in the markup-view
- *          - {InspectorPanel} The instance of the InspectorPanel opened
- * @param {DOMNode|String} nodeOrSelector The node or node selector
- * corresponding to the test element
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently
- * opened
- */
-function* runAddAttributesTest(test, nodeOrSelector, inspector) {
-  let element = getNode(nodeOrSelector);
-
-  info("Starting add-attribute test: " + test.desc);
-  yield addNewAttributes(element, test.text, inspector);
-
-  info("Assert that the attribute(s) has/have been applied correctly");
-  assertAttributes(element, test.expectedAttributes);
-
-  if (test.validate) {
-    test.validate(element, getContainerForRawNode(element, inspector), inspector);
-  }
-
-  info("Undo the change");
-  yield undoChange(inspector);
-
-  info("Assert that the attribute(s) has/have been removed correctly");
-  assertAttributes(element, {});
-}
-
-/**
- * Run a series of edit-attributes tests.
- * This function will iterate over the provided tests array and run each test.
- * Each test's goal is to locate a given element on the current test page, assert
- * its current attributes, then provide the name of one of them and a value to
- * be set into it, and then check if the new attributes are correct.
- * After each test has run, the markup-view's undo and redo commands will be
- * called and the test runner will assert again that the attributes are correct.
- * @param {Array} tests See runEditAttributesTest for the structure
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently
- * opened
- * @return a promise that resolves when the tests have run
- */
-function runEditAttributesTests(tests, inspector) {
-  info("Running " + tests.length + " edit-attributes tests");
-  return Task.spawn(function*() {
-    info("Expanding all nodes in the markup-view");
-    yield inspector.markup.expandAll();
-
-    for (let test of tests) {
-      yield runEditAttributesTest(test, inspector);
-    }
-
-    yield inspector.once("inspector-updated");
-  });
-}
-
-/**
- * Run a single edit-attribute test.
- * See runEditAttributesTests for a description.
- * @param {Object} test A test object should contain the following properties:
- *        - desc {String} a textual description for that test, to help when
- *        reading logs
- *        - node {String} a css selector that will be used to select the node
- *        which will be tested during this iteration
- *        - originalAttributes {Object} a key/value pair object that will be
- *        used to check the attributes of the node before the test runs
- *        - name {String} the name of the attribute to focus the editor for
- *        - value {String} the new value to be typed in the focused editor
- *        - expectedAttributes {Object} a key/value pair object that will be
- *        used to check the attributes on the test element
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently
- * opened
- */
-function* runEditAttributesTest(test, inspector) {
-  info("Starting edit-attribute test: " + test.desc);
-
-  info("Selecting the test node " + test.node);
-  yield selectNode(test.node, inspector);
-
-  info("Asserting that the node has the right attributes to start with");
-  assertAttributes(test.node, test.originalAttributes);
-
-  info("Editing attribute " + test.name + " with value " + test.value);
-
-  let container = getContainerForRawNode(test.node, inspector);
-  ok(container && container.editor, "The markup-container for " + test.node +
-    " was found");
-
-  info("Listening for the markupmutation event");
-  let nodeMutated = inspector.once("markupmutation");
-  let attr = container.editor.attrs[test.name].querySelector(".editable");
-  setEditableFieldValue(attr, test.value, inspector);
-  yield nodeMutated;
-
-  info("Asserting the new attributes after edition");
-  assertAttributes(test.node, test.expectedAttributes);
-
-  info("Undo the change and assert that the attributes have been changed back");
-  yield undoChange(inspector);
-  assertAttributes(test.node, test.originalAttributes);
-
-  info("Redo the change and assert that the attributes have been changed again");
-  yield redoChange(inspector);
-  assertAttributes(test.node, test.expectedAttributes);
-}
-
-/**
- * Run a series of edit-outer-html tests.
- * This function will iterate over the provided tests array and run each test.
- * Each test's goal is to provide a node (a selector) and a new outer-HTML to be
- * inserted in place of the current one for that node.
- * This test runner will wait for the mutation event to be fired and will check
- * a few things. Each test may also provide its own validate function to perform
- * assertions and verify that the new outer html is correct.
- * @param {Array} tests See runEditOuterHTMLTest for the structure
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently
- * opened
- * @return a promise that resolves when the tests have run
- */
-function runEditOuterHTMLTests(tests, inspector) {
-  info("Running " + tests.length + " edit-outer-html tests");
-  return Task.spawn(function* () {
-    for (let step of TEST_DATA) {
-      yield runEditOuterHTMLTest(step, inspector);
-    }
-  });
-}
-
-/**
- * Run a single edit-outer-html test.
- * See runEditOuterHTMLTests for a description.
- * @param {Object} test A test object should contain the following properties:
- *        - selector {String} a css selector targeting the node to edit
- *        - oldHTML {String}
- *        - newHTML {String}
- *        - validate {Function} will be executed when the edition test is done,
- *        after the new outer-html has been inserted. Should be used to verify
- *        the actual DOM, see if it corresponds to the newHTML string provided
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently
- * opened
- */
-function* runEditOuterHTMLTest(test, inspector) {
-  info("Running an edit outerHTML test on '" + test.selector + "'");
-  yield selectNode(test.selector, inspector);
-  let oldNodeFront = inspector.selection.nodeFront;
-
-  info("Listening for the markupmutation event");
-  // This event fires once the outerHTML is set, with a target as the parent node and a type of "childList".
-  let mutated = inspector.once("markupmutation");
-  info("Editing the outerHTML");
-  inspector.markup.updateNodeOuterHTML(inspector.selection.nodeFront, test.newHTML, test.oldHTML);
-  let mutations = yield mutated;
-  ok(true, "The markupmutation event has fired, mutation done");
-
-  info("Check to make the sure the correct mutation event was fired, and that the parent is selected");
-  let nodeFront = inspector.selection.nodeFront;
-  let mutation = mutations[0];
-  let isFromOuterHTML = mutation.removed.some(n => n === oldNodeFront);
-
-  ok(isFromOuterHTML, "The node is in the 'removed' list of the mutation");
-  is(mutation.type, "childList", "Mutation is a childList after updating outerHTML");
-  is(mutation.target, nodeFront, "Parent node is selected immediately after setting outerHTML");
-
-  // Wait for node to be reselected after outerHTML has been set
-  yield inspector.selection.once("new-node");
-
-  // Typically selectedNode will === pageNode, but if a new element has been injected in front
-  // of it, this will not be the case.  If this happens.
-  let selectedNode = inspector.selection.node;
-  let nodeFront = inspector.selection.nodeFront;
-  let pageNode = getNode(test.selector);
-
-  if (test.validate) {
-    test.validate(pageNode, selectedNode);
-  } else {
-    is(pageNode, selectedNode, "Original node (grabbed by selector) is selected");
-    is(pageNode.outerHTML, test.newHTML, "Outer HTML has been updated");
-  }
-
-  // Wait for the inspector to be fully updated to avoid causing errors by
-  // abruptly closing hanging requests when the test ends
-  yield inspector.once("inspector-updated");
-}
new file mode 100644
--- /dev/null
+++ b/browser/devtools/markupview/test/helper_attributes_test_runner.js
@@ -0,0 +1,151 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+/**
+ * Run a series of add-attributes tests.
+ * This function will iterate over the provided tests array and run each test.
+ * Each test's goal is to provide some text to be entered into the test node's
+ * new-attribute field and check that the given attributes have been created.
+ * After each test has run, the markup-view's undo command will be called and
+ * the test runner will check if all the new attributes are gone.
+ * @param {Array} tests See runAddAttributesTest for the structure
+ * @param {DOMNode|String} nodeOrSelector The node or node selector
+ * corresponding to an element on the current test page that has *no attributes*
+ * when the test starts. It will be used to add and remove attributes.
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * opened
+ * @return a promise that resolves when the tests have run
+ */
+function runAddAttributesTests(tests, nodeOrSelector, inspector) {
+  info("Running " + tests.length + " add-attributes tests");
+  return Task.spawn(function*() {
+    info("Selecting the test node");
+    let div = getNode("div");
+    yield selectNode(div, inspector);
+
+    for (let test of tests) {
+      yield runAddAttributesTest(test, div, inspector);
+    }
+
+    yield inspector.once("inspector-updated");
+  });
+}
+
+/**
+ * Run a single add-attribute test.
+ * See runAddAttributesTests for a description.
+ * @param {Object} test A test object should contain the following properties:
+ *        - desc {String} a textual description for that test, to help when
+ *        reading logs
+ *        - text {String} the string to be inserted into the new attribute field
+ *        - expectedAttributes {Object} a key/value pair object that will be
+ *        used to check the attributes on the test element
+ *        - validate {Function} optional extra function that will be called after
+ *        the attributes have been added and which should be used to assert some
+ *        more things this test runner might not be checking. The function will
+ *        be called with the following arguments:
+ *          - {DOMNode} The element being tested
+ *          - {MarkupContainer} The corresponding container in the markup-view
+ *          - {InspectorPanel} The instance of the InspectorPanel opened
+ * @param {DOMNode|String} nodeOrSelector The node or node selector
+ * corresponding to the test element
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * opened
+ */
+function* runAddAttributesTest(test, nodeOrSelector, inspector) {
+  let element = getNode(nodeOrSelector);
+
+  info("Starting add-attribute test: " + test.desc);
+  yield addNewAttributes(element, test.text, inspector);
+
+  info("Assert that the attribute(s) has/have been applied correctly");
+  assertAttributes(element, test.expectedAttributes);
+
+  if (test.validate) {
+    test.validate(element, getContainerForRawNode(element, inspector), inspector);
+  }
+
+  info("Undo the change");
+  yield undoChange(inspector);
+
+  info("Assert that the attribute(s) has/have been removed correctly");
+  assertAttributes(element, {});
+}
+
+/**
+ * Run a series of edit-attributes tests.
+ * This function will iterate over the provided tests array and run each test.
+ * Each test's goal is to locate a given element on the current test page, assert
+ * its current attributes, then provide the name of one of them and a value to
+ * be set into it, and then check if the new attributes are correct.
+ * After each test has run, the markup-view's undo and redo commands will be
+ * called and the test runner will assert again that the attributes are correct.
+ * @param {Array} tests See runEditAttributesTest for the structure
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * opened
+ * @return a promise that resolves when the tests have run
+ */
+function runEditAttributesTests(tests, inspector) {
+  info("Running " + tests.length + " edit-attributes tests");
+  return Task.spawn(function*() {
+    info("Expanding all nodes in the markup-view");
+    yield inspector.markup.expandAll();
+
+    for (let test of tests) {
+      yield runEditAttributesTest(test, inspector);
+    }
+
+    yield inspector.once("inspector-updated");
+  });
+}
+
+/**
+ * Run a single edit-attribute test.
+ * See runEditAttributesTests for a description.
+ * @param {Object} test A test object should contain the following properties:
+ *        - desc {String} a textual description for that test, to help when
+ *        reading logs
+ *        - node {String} a css selector that will be used to select the node
+ *        which will be tested during this iteration
+ *        - originalAttributes {Object} a key/value pair object that will be
+ *        used to check the attributes of the node before the test runs
+ *        - name {String} the name of the attribute to focus the editor for
+ *        - value {String} the new value to be typed in the focused editor
+ *        - expectedAttributes {Object} a key/value pair object that will be
+ *        used to check the attributes on the test element
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * opened
+ */
+function* runEditAttributesTest(test, inspector) {
+  info("Starting edit-attribute test: " + test.desc);
+
+  info("Selecting the test node " + test.node);
+  yield selectNode(test.node, inspector);
+
+  info("Asserting that the node has the right attributes to start with");
+  assertAttributes(test.node, test.originalAttributes);
+
+  info("Editing attribute " + test.name + " with value " + test.value);
+
+  let container = getContainerForRawNode(test.node, inspector);
+  ok(container && container.editor, "The markup-container for " + test.node +
+    " was found");
+
+  info("Listening for the markupmutation event");
+  let nodeMutated = inspector.once("markupmutation");
+  let attr = container.editor.attrs[test.name].querySelector(".editable");
+  setEditableFieldValue(attr, test.value, inspector);
+  yield nodeMutated;
+
+  info("Asserting the new attributes after edition");
+  assertAttributes(test.node, test.expectedAttributes);
+
+  info("Undo the change and assert that the attributes have been changed back");
+  yield undoChange(inspector);
+  assertAttributes(test.node, test.originalAttributes);
+
+  info("Redo the change and assert that the attributes have been changed again");
+  yield redoChange(inspector);
+  assertAttributes(test.node, test.expectedAttributes);
+}
new file mode 100644
--- /dev/null
+++ b/browser/devtools/markupview/test/helper_outerhtml_test_runner.js
@@ -0,0 +1,81 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+/**
+ * Run a series of edit-outer-html tests.
+ * This function will iterate over the provided tests array and run each test.
+ * Each test's goal is to provide a node (a selector) and a new outer-HTML to be
+ * inserted in place of the current one for that node.
+ * This test runner will wait for the mutation event to be fired and will check
+ * a few things. Each test may also provide its own validate function to perform
+ * assertions and verify that the new outer html is correct.
+ * @param {Array} tests See runEditOuterHTMLTest for the structure
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * opened
+ * @return a promise that resolves when the tests have run
+ */
+function runEditOuterHTMLTests(tests, inspector) {
+  info("Running " + tests.length + " edit-outer-html tests");
+  return Task.spawn(function* () {
+    for (let step of TEST_DATA) {
+      yield runEditOuterHTMLTest(step, inspector);
+    }
+  });
+}
+
+/**
+ * Run a single edit-outer-html test.
+ * See runEditOuterHTMLTests for a description.
+ * @param {Object} test A test object should contain the following properties:
+ *        - selector {String} a css selector targeting the node to edit
+ *        - oldHTML {String}
+ *        - newHTML {String}
+ *        - validate {Function} will be executed when the edition test is done,
+ *        after the new outer-html has been inserted. Should be used to verify
+ *        the actual DOM, see if it corresponds to the newHTML string provided
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * opened
+ */
+function* runEditOuterHTMLTest(test, inspector) {
+  info("Running an edit outerHTML test on '" + test.selector + "'");
+  yield selectNode(test.selector, inspector);
+  let oldNodeFront = inspector.selection.nodeFront;
+
+  info("Listening for the markupmutation event");
+  // This event fires once the outerHTML is set, with a target as the parent node and a type of "childList".
+  let mutated = inspector.once("markupmutation");
+  info("Editing the outerHTML");
+  inspector.markup.updateNodeOuterHTML(inspector.selection.nodeFront, test.newHTML, test.oldHTML);
+  let mutations = yield mutated;
+  ok(true, "The markupmutation event has fired, mutation done");
+
+  info("Check to make the sure the correct mutation event was fired, and that the parent is selected");
+  let nodeFront = inspector.selection.nodeFront;
+  let mutation = mutations[0];
+  let isFromOuterHTML = mutation.removed.some(n => n === oldNodeFront);
+
+  ok(isFromOuterHTML, "The node is in the 'removed' list of the mutation");
+  is(mutation.type, "childList", "Mutation is a childList after updating outerHTML");
+  is(mutation.target, nodeFront, "Parent node is selected immediately after setting outerHTML");
+
+  // Wait for node to be reselected after outerHTML has been set
+  yield inspector.selection.once("new-node");
+
+  // Typically selectedNode will === pageNode, but if a new element has been injected in front
+  // of it, this will not be the case.  If this happens.
+  let selectedNode = inspector.selection.node;
+  let nodeFront = inspector.selection.nodeFront;
+  let pageNode = getNode(test.selector);
+
+  if (test.validate) {
+    test.validate(pageNode, selectedNode);
+  } else {
+    is(pageNode, selectedNode, "Original node (grabbed by selector) is selected");
+    is(pageNode.outerHTML, test.newHTML, "Outer HTML has been updated");
+  }
+
+  // Wait for the inspector to be fully updated to avoid causing errors by
+  // abruptly closing hanging requests when the test ends
+  yield inspector.once("inspector-updated");
+}