<!DOCTYPE html>
<html>
  <head>
    <title>Plugin instantiation</title>
    <script type="application/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
    <script type="application/javascript" src="/tests/SimpleTest/SpecialPowers.js"></script>
    <script type="application/javascript" src="plugin-utils.js"></script>
    <meta charset="utf-8">
  <body onload="onLoad()">
    <script class="testbody" type="text/javascript;version=1.8">

      "use strict";
      SimpleTest.waitForExplicitFinish();

      setTestPluginEnabledState(SpecialPowers.Ci.nsIPluginTag.STATE_ENABLED, "Test Plug-in");

      // This can go away once embed also is on WebIDL
      let OBJLC = SpecialPowers.Ci.nsIObjectLoadingContent;

      // Use string modes in this test to make the test easier to read/debug.
      // nsIObjectLoadingContent refers to this as "type", but I am using "mode"
      // in the test to avoid confusing with content-type.
      let prettyModes = {};
      prettyModes[OBJLC.TYPE_LOADING] = "loading";
      prettyModes[OBJLC.TYPE_IMAGE] = "image";
      prettyModes[OBJLC.TYPE_PLUGIN] = "plugin";
      prettyModes[OBJLC.TYPE_DOCUMENT] = "document";
      prettyModes[OBJLC.TYPE_NULL] = "none";

      let body = document.body;
      // A single-pixel white png
      let testPNG = '';
      // An empty, but valid, SVG
      let testSVG = 'data:image/svg+xml,<?xml version="1.0" encoding="UTF-8"?><svg xmlns="http://www.w3.org/2000/svg" width="100" height="100"></svg>';
      // executeSoon wrapper to count pending callbacks
      let pendingCalls = 0;
      let afterPendingCalls = false;

      function runWhenDone(func) {
        if (!pendingCalls)
          func();
        else
          afterPendingCalls = func;
      }
      function runSoon(func) {
        pendingCalls++;
        SimpleTest.executeSoon(function() {
          func();
          if (--pendingCalls < 1 && afterPendingCalls)
            afterPendingCalls();
        });
      }
      function src(obj, state, uri) {
        // If we have a running plugin, src changing should always throw it out,
        // even if it causes us to load the same plugin again.
        if (uri && runningPlugin(obj, state)) {
          if (!state.oldPlugins)
            state.oldPlugins = [];
          try {
            state.oldPlugins.push(obj.getObjectValue());
          } catch (e) {
            ok(false, "Running plugin but cannot call getObjectValue?");
          }
        }

        var srcattr;
        if (state.tagName == "object")
          srcattr = "data";
        else if (state.tagName == "embed")
          srcattr = "src";
        else
          ok(false, "Internal test fail: Why are we setting the src of an applet");

        // Plugins should always go away immediately on src/data change
        state.initialPlugin = false;
        if (uri === null) {
          removeAttr(obj, srcattr);
          // TODO Bug 767631 - we don't trigger loadObject on UnsetAttr :(
          forceReload(obj, state);
        } else {
          setAttr(obj, srcattr, uri);
        }
      }
      // We have to be careful not to reach past the nsObjectLoadingContent
      // prototype to touch generic element attributes, as this will try to
      // spawn the plugin, breaking our ability to test for that.
      function getAttr(obj, attr) {
        return document.body.constructor.prototype.getAttribute.call(obj, attr);
      }
      function setAttr(obj, attr, val) {
        return document.body.constructor.prototype.setAttribute.call(obj, attr, val);
      }
      function hasAttr(obj, attr) {
        return document.body.constructor.prototype.hasAttribute.call(obj, attr);
      }
      function removeAttr(obj, attr) {
        return document.body.constructor.prototype.removeAttribute.call(obj, attr);
      }
      function setDisplayed(obj, display) {
        if (display)
          removeAttr(obj, 'style');
        else
          setAttr(obj, 'style', "display: none;");
      }
      function displayed(obj) {
        // Hacky, but that's all we use style for.
        return !hasAttr(obj, 'style');
      }
      function actualType(obj, state) {
        return state.getActualType.call(obj);
      }
      function getMode(obj, state) {
        return prettyModes[state.getDisplayedType.call(obj)];
      }
      function runningPlugin(obj, state) {
        return state.getHasRunningPlugin.call(obj);
      }

      // TODO this is hacky and might hide some failures, but is needed until
      // Bug 767635 lands -- which itself will probably mean tweaking this test.
      function forceReload(obj, state) {
        let attr;
        if (state.tagName == "object")
          attr = "data";
        else if (state.tagName == "embed")
          attr = "src";

        if (attr && hasAttr(obj, attr)) {
          src(obj, state, getAttr(obj, attr));
        } else if (body.contains(obj)) {
          body.appendChild(obj);
        } else {
          // Out of document nodes without data attributes simply can't be
          // reloaded currently. Bug 767635
        }
      };

      // Make a list of combinations of sub-lists, e.g.:
      // [ [a, b], [c, d] ]
      // ->
      // [ [a, c], [a, d], [b, c], [b, d] ]
      function eachList() {
        let all = [];
        if (!arguments.length)
          return all;
        let list = Array.prototype.slice.call(arguments, 0);
        for (let c of list[0]) {
          if (list.length > 1) {
            for (let x of eachList.apply(this,list.slice(1))) {
              all.push((c.length ? [c] : []).concat(x));
            }
          } else if (c.length) {
            all.push([c]);
          }
        }
        return all;
      }

      let states = {
        svg: function(obj, state) {
          removeAttr(obj, "type");
          src(obj, state, testSVG);
          state.noChannel = false;
          state.expectedType = "image/svg";
          // SVGs are actually image-like subdocuments
          state.expectedMode = "document";
        },
        image: function(obj, state) {
          removeAttr(obj, "type");
          src(obj, state, testPNG);
          state.noChannel = false;
          state.expectedMode = "image";
          state.expectedType = "image/png";
        },
        plugin: function(obj, state) {
          removeAttr(obj, "type");
          src(obj, state, "data:application/x-test,foo");
          state.noChannel = false;
          state.expectedType = "application/x-test";
          state.expectedMode = "plugin";
        },
        pluginExtension: function(obj, state) {
          src(obj, state, "./fake_plugin.tst");
          state.expectedMode = "plugin";
          state.pluginExtension = true;
          state.noChannel = false;
        },
        document: function(obj, state) {
          removeAttr(obj, "type");
          src(obj, state, "data:text/plain,I am a document");
          state.noChannel = false;
          state.expectedType = "text/plain";
          state.expectedMode = "document";
        },
        fallback: function(obj, state) {
          removeAttr(obj, "type");
          state.expectedType = "application/x-unknown";
          state.expectedMode = "none";
          state.noChannel = true;
          src(obj, state, null);
        },
        addToDoc: function(obj, state) {
          body.appendChild(obj);
        },
        removeFromDoc: function(obj, state) {
          if (body.contains(obj))
            body.removeChild(obj);
        },
        // Set the proper type
        setType: function(obj, state) {
          if (state.expectedType) {
            state.badType = false;
            setAttr(obj, 'type', state.expectedType);
            forceReload(obj, state);
          }
        },
        // Set an improper type
        setWrongType: function(obj, state) {
          // This should break no-channel-plugins but nothing else
          state.badType = true;
          setAttr(obj, 'type', "application/x-unknown");
          forceReload(obj, state);
        },
        // Set a plugin type
        setPluginType: function(obj, state) {
          // If an object/embed has a type set to a plugin type, it should not
          // use the channel type.
          state.badType = false;
          setAttr(obj, 'type', 'application/x-test');
          state.expectedType = "application/x-test";
          state.expectedMode = "plugin";
          forceReload(obj, state);
        },
        noChannel: function(obj, state) {
          src(obj, state, null);
          state.noChannel = true;
          state.pluginExtension = false;
        },
        displayNone: function(obj, state) {
          setDisplayed(obj, false);
        },
        displayInherit: function(obj, state) {
          setDisplayed(obj, true);
        }
      };


      function testObject(obj, state) {
        // If our test combination both sets noChannel but no explicit type
        // it shouldn't load ever.
        let expectedMode = state.expectedMode;
        let actualMode = getMode(obj, state);

        if (state.noChannel && !getAttr(obj, 'type')) {
          // Some combinations of test both set no type and no channel. This is
          // worth testing with the various combinations, but shouldn't load.
          expectedMode = "none";
        }

        // Embed tags should always try to load a plugin by type or extension
        // before falling back to opening a channel. See bug 803159
        if (state.tagName == "embed" &&
            (getAttr(obj, 'type') == "application/x-test" || state.pluginExtension)) {
          state.noChannel = true;
        }

        // with state.loading, unless we're loading with no channel, these types
        // should still be in loading state pending a channel.
        if (state.loading && (expectedMode == "image" || expectedMode == "document" ||
                             (expectedMode == "plugin" && !state.initialPlugin && !state.noChannel))) {
          expectedMode = "loading";
        }

        // With the exception of plugins with a proper type, nothing should
        // load without a channel
        if (state.noChannel && (expectedMode != "plugin" || state.badType) &&
            body.contains(obj)) {
          expectedMode = "none";
        }

        // embed tags should reject documents, except for SVG images which
        // render as such
        if (state.tagName == "embed" && expectedMode == "document" &&
            actualType(obj, state) != "image/svg+xml") {
          expectedMode = "none";
        }

        // Embeds with a plugin type should skip opening a channel prior to
        // loading, taking only type into account.
        if (state.tagName == 'embed' && getAttr(obj, 'type') == 'application/x-test' &&
            body.contains(obj)) {
          expectedMode = "plugin";
        }

        if (!body.contains(obj)
            && (!state.loading || expectedMode != "image")
            && (!state.initialPlugin || expectedMode != "plugin")) {
          // Images are handled by nsIImageLoadingContent so we dont track
          // their state change as they're detached and reattached. All other
          // types switch to state "loading", and are completely unloaded
          expectedMode = "loading";
        }

        is(actualMode, expectedMode, "check loaded mode");

        // If we're a plugin, check that we spawned successfully. state.loading
        // is set if we haven't had an event loop since applying state, in which
        // case the plugin would not have stopped yet if it was initially a
        // plugin.
        let shouldBeSpawnable = expectedMode == "plugin" && displayed(obj);
        let shouldSpawn = shouldBeSpawnable && (!state.loading || state.initialPlugin);
        let didSpawn = runningPlugin(obj, state);
        is(didSpawn, !!shouldSpawn, "check plugin spawned is " + !!shouldSpawn);

        // If we are a plugin, scripting should work. If we're not spawned we
        // should spawn synchronously.
        let scripted = false;
        try {
          let x = obj.getObjectValue();
          scripted = true;
        } catch(e) {}
        is(scripted, shouldBeSpawnable, "check plugin scriptability");

        // If this tag previously had other spawned plugins, make sure it
        // respawned between then and now
        if (state.oldPlugins && didSpawn) {
          let didRespawn = false;
          for (let oldp of state.oldPlugins) {
            // If this returns false or throws, it's not the same plugin
            try {
              didRespawn = !obj.checkObjectValue(oldp);
            } catch (e) {
              didRespawn = true;
            }
          }
          is(didRespawn, true, "Plugin should have re-spawned since old state ("+state.oldPlugins.length+")");
        }
      }

      let total = 0;
      let test_modes = {
        // Just apply from_state then to_state
        "immediate": function(obj, from_state, to_state, state) {
          for (let from of from_state)
            states[from](obj, state);
          for (let to of to_state)
            states[to](obj, state);

          // We don't spin the event loop between applying to_state and
          // running tests, so some types are still loading
          state.loading = true;
          info("["+(++total)+"] Testing [ " + from_state + " ] -> [ " + to_state + " ] / " + state.tagName + " / immediate");
          testObject(obj, state);

          if (body.contains(obj))
            body.removeChild(obj);

        },
        // Apply states, spin event loop, run tests.
        "cycle": function(obj, from_state, to_state, state) {
          for (let from of from_state)
            states[from](obj, state);
          for (let to of to_state)
            states[to](obj, state);
          // Because re-appending to the document creates a script blocker, but
          // plugins spawn asynchronously, we need to return to the event loop
          // twice to ensure the plugin has been given a chance to lazily spawn.
          runSoon(function() { runSoon(function() {
            info("["+(++total)+"] Testing [ " + from_state + " ] -> [ " + to_state + " ] / " + state.tagName + " / cycle");
            testObject(obj, state);

            if (body.contains(obj))
              body.removeChild(obj);
          }); });
        },
        // Apply initial state, spin event loop, apply final state, spin event
        // loop again.
        "cycleboth": function(obj, from_state, to_state, state) {
          for (let from of from_state) {
            states[from](obj, state);
          }
          runSoon(function() {
            for (let to of to_state) {
              states[to](obj, state);
            }
            // Because re-appending to the document creates a script blocker,
            // but plugins spawn asynchronously, we need to return to the event
            // loop twice to ensure the plugin has been given a chance to lazily
            // spawn.
            runSoon(function() { runSoon(function() {
              info("["+(++total)+"] Testing [ " + from_state + " ] -> [ " + to_state + " ] / " + state.tagName + " / cycleboth");
              testObject(obj, state);

              if (body.contains(obj))
                body.removeChild(obj);
            }); });
          });
        },
        // Apply initial state, spin event loop, apply later state, test
        // immediately
        "cyclefirst": function(obj, from_state, to_state, state) {
          for (let from of from_state) {
            states[from](obj, state);
          }
          runSoon(function() {
            state.initialPlugin = runningPlugin(obj, state);
            for (let to of to_state) {
              states[to](obj, state);
            }
            info("["+(++total)+"] Testing [ " + from_state + " ] -> [ " + to_state + " ] / " + state.tagName + " / cyclefirst");
            // We don't spin the event loop between applying to_state and
            // running tests, so some types are still loading
            state.loading = true;
            testObject(obj, state);

            if (body.contains(obj))
              body.removeChild(obj);
          });
        },
      };

      function test(testdat) {
        // FIXME bug 1291854: Change back to lets when the test is fixed.
        for (var from_state of testdat['from_states']) {
          for (var to_state of testdat['to_states']) {
            for (var mode of testdat['test_modes']) {
              for (var type of testdat['tag_types']) {
                runSoon(function () {
                  let obj = document.createElement(type);
                  obj.width = 1; obj.height = 1;
                  let state = {};
                  state.noChannel = true;
                  state.tagName = type;
                  // Part of the test checks whether a plugin spawned or not,
                  // but touching the object prototype will attempt to
                  // synchronously spawn a plugin!  We use this terrible hack to
                  // get a privileged getter for the attributes we want to touch
                  // prior to applying any attributes.
                  // TODO when embed goes away we wont need to check for
                  //      QueryInterface any longer.
                  var lookup_on = obj.QueryInterface ? obj.QueryInterface(OBJLC): obj;
                  state.getDisplayedType = SpecialPowers.do_lookupGetter(lookup_on, 'displayedType');
                  state.getHasRunningPlugin = SpecialPowers.do_lookupGetter(lookup_on, 'hasRunningPlugin');
                  state.getActualType = SpecialPowers.do_lookupGetter(lookup_on, 'actualType');
                  test_modes[mode](obj, from_state, to_state, state);
                });
              }
            }
          }
        }
      }

      function onLoad() {
        // Generic tests
        test({
          'tag_types': [ 'embed', 'object' ],
          // In all three modes
          'test_modes': [ 'immediate', 'cycle', 'cyclefirst', 'cycleboth' ],
          // Starting from a blank tag in and out of the document, a loading
          // plugin, and no-channel plugin (initial types only really have
          // odd cases with plugins)
          'from_states': [
            [ 'addToDoc' ],
            [ 'plugin' ],
            [ 'plugin', 'addToDoc' ],
            [ 'plugin', 'noChannel', 'setType', 'addToDoc' ],
            [],
          ],
          // To various combinations of loaded objects
          'to_states': eachList(
            [ 'svg', 'image', 'plugin', 'document', '' ],
            [ 'setType', 'setWrongType', 'setPluginType', '' ],
            [ 'noChannel', '' ],
            [ 'displayNone', 'displayInherit', '' ]
          )});
        // Special case test for embed tags with plugin-by-extension
        // TODO object tags should be tested here too -- they have slightly
        //      different behavior, but waiting on a file load requires a loaded
        //      event handler and wont work with just our event loop spinning.
        test({
          'tag_types': [ 'embed' ],
          'test_modes': [ 'immediate', 'cyclefirst', 'cycle', 'cycleboth' ],
          'from_states': eachList(
            [ 'svg', 'plugin', 'image', 'document' ],
            [ 'addToDoc' ]
          ),
          // Set extension along with valid ty
          'to_states': [
            [ 'pluginExtension' ]
          ]});
        // Test plugin add/remove from document with adding/removing frame, with
        // and without a channel.
        test({
          'tag_types': [ 'embed', 'object' ], // Ideally we'd test object too, but this gets exponentially long.
          'test_modes': [ 'immediate', 'cyclefirst', 'cycle' ],
          'from_states': [ [ 'displayNone', 'plugin', 'addToDoc' ],
                           [ 'displayNone', 'plugin', 'noChannel', 'addToDoc' ],
                           [ 'plugin', 'noChannel', 'addToDoc' ],
                           [ 'plugin', 'noChannel' ] ],
          'to_states': eachList(
            [ 'displayNone', '' ],
            [ 'removeFromDoc' ],
            [ 'image', 'displayNone', '' ],
            [ 'image', 'displayNone', '' ],
            [ 'addToDoc' ],
            [ 'displayInherit' ]
          )});
        runWhenDone(() => SimpleTest.finish());
      }
    </script>