/* Any copyright is dedicated to the Public Domain.
   http://creativecommons.org/publicdomain/zero/1.0/ */

"use strict";

importScripts("utils_worker.js"); // Test suite code
info("Test suite configured");

importScripts("resource://gre/modules/workers/require.js");
info("Loader imported");

var PATH = "chrome://mochitests/content/chrome/toolkit/components/workerloader/tests/";
var tests = [];
var add_test = function(test) {
  tests.push(test);
};

add_test(function test_setup() {
  ok(typeof require != "undefined", "Function |require| is defined");
});

// Test simple loading (moduleA-depends.js requires moduleB-dependency.js)
add_test(function test_load() {
  let A = require(PATH + "moduleA-depends.js");
  ok(true, "Opened module A");

  is(A.A, true, "Module A exported value A");
  ok(!("B" in A), "Module A did not export value B");
  is(A.importedFoo, "foo", "Module A re-exported B.foo");

  // re-evaluating moduleB-dependency.js would cause an error, but re-requiring it shouldn't
  let B = require(PATH + "moduleB-dependency.js");
  ok(true, "Managed to re-require module B");
  is(B.B, true, "Module B exported value B");
  is(B.foo, "foo", "Module B exported value foo");
});

// Test simple circular loading (moduleC-circular.js and moduleD-circular.js require each other)
add_test(function test_circular() {
  let C = require(PATH + "moduleC-circular.js");
  ok(true, "Loaded circular modules C and D");
  is(C.copiedFromD.copiedFromC.enteredC, true, "Properties exported by C before requiring D can be seen by D immediately");

  let D = require(PATH + "moduleD-circular.js");
  is(D.exportedFromC.finishedC, true, "Properties exported by C after requiring D can be seen by D eventually");
});

// Testing error cases
add_test(function test_exceptions() {
  let should_throw = function(f) {
    try {
      f();
      return null;
    } catch (ex) {
      return ex;
    }
  };

  let exn = should_throw(() => require(PATH + "this module doesn't exist"));
  ok(!!exn, "Attempting to load a module that doesn't exist raises an error");

  exn = should_throw(() => require(PATH + "moduleE-throws-during-require.js"));
  ok(!!exn, "Attempting to load a module that throws at toplevel raises an error");
  is(exn.moduleName, PATH + "moduleE-throws-during-require.js",
    "moduleName is correct");
  isnot(exn.moduleStack.indexOf("moduleE-throws-during-require.js"), -1,
    "moduleStack contains the name of the module");
  is(exn.lineNumber, 10, "The error comes with the right line number");

  exn = should_throw(() => require(PATH + "moduleF-syntaxerror.xml"));
  ok(!!exn, "Attempting to load a non-well formatted module raises an error");

  exn = should_throw(() => require(PATH + "moduleG-throws-later.js").doThrow());
  ok(!!exn, "G.doThrow() has raised an error");
  info(exn);
  ok(exn.toString().startsWith("TypeError"), "The exception is a TypeError.");
  is(exn.moduleName, PATH + "moduleG-throws-later.js", "The name of the module is correct");
  isnot(exn.moduleStack.indexOf("moduleG-throws-later.js"), -1,
    "The name of the right file appears somewhere in the stack");
  is(exn.lineNumber, 11, "The error comes with the right line number");
});

function get_exn(f) {
  try {
    f();
    return undefined;
  } catch (ex) {
    return ex;
  }
}

// Test module.exports
add_test(function test_module_dot_exports() {
  let H = require(PATH + "moduleH-module-dot-exports.js");
  is(H.key, "value", "module.exports worked");
  let H2 = require(PATH + "moduleH-module-dot-exports.js");
  is(H2.key, "value", "module.exports returned the same key");
  ok(H2 === H, "module.exports returned the same module the second time");
  let exn = get_exn(() => H.key = "this should not be accepted");
  ok(exn instanceof TypeError, "Cannot alter value in module.exports after export");
  exn = get_exn(() => H.key2 = "this should not be accepted, either");
  ok(exn instanceof TypeError, "Cannot add value to module.exports after export");
});

self.onmessage = function(message) {
  for (let test of tests) {
    info("Entering " + test.name);
    try {
      test();
    } catch (ex) {
      ok(false, "Test " + test.name + " failed");
      info(ex);
      info(ex.stack);
    }
    info("Leaving " + test.name);
  }
  finish();
};