summaryrefslogtreecommitdiffstats
path: root/addon-sdk/source/test/test-system-input-output.js
diff options
context:
space:
mode:
Diffstat (limited to 'addon-sdk/source/test/test-system-input-output.js')
-rw-r--r--addon-sdk/source/test/test-system-input-output.js319
1 files changed, 319 insertions, 0 deletions
diff --git a/addon-sdk/source/test/test-system-input-output.js b/addon-sdk/source/test/test-system-input-output.js
new file mode 100644
index 000000000..9a4d6ebb9
--- /dev/null
+++ b/addon-sdk/source/test/test-system-input-output.js
@@ -0,0 +1,319 @@
+/* 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/. */
+"use strict";
+
+
+const { id: addonID, name: addonName } = require("sdk/self");
+const { Cc, Ci, Cu } = require("chrome");
+const { Loader, LoaderWithHookedConsole2 } = require("sdk/test/loader");
+const { InputPort } = require("sdk/input/system");
+const { OutputPort } = require("sdk/output/system");
+
+const { removeObserver, addObserver,
+ notifyObservers } = Cc["@mozilla.org/observer-service;1"].
+ getService(Ci.nsIObserverService);
+
+const { lift, start, stop, send } = require("sdk/event/utils");
+
+const isConsoleEvent = topic =>
+ ["console-api-log-event",
+ "console-storage-cache-event"].indexOf(topic) >= 0;
+
+const message = x => ({wrappedJSObject: {data: x}});
+
+exports["test start / stop ports"] = assert => {
+ const input = new InputPort({ id: Date.now().toString(32), initial: {data:0} });
+ const topic = input.topic;
+
+ assert.ok(topic.includes(addonID), "topics are namespaced to add-on");
+
+ const xs = lift(({data}) => "x:" + data, input);
+ const ys = lift(({data}) => "y:" + data, input);
+
+ assert.deepEqual(input.value, {data:0}, "initila value is set");
+ assert.deepEqual(xs.value, "x:0", "initial value is mapped");
+ assert.deepEqual(ys.value, "y:0", "initial value is mapped");
+
+ notifyObservers(message(1), topic, null);
+
+ assert.deepEqual(input.value, {data:0}, "no message received on input port");
+ assert.deepEqual(xs.value, "x:0", "no message received on xs");
+ assert.deepEqual(ys.value, "y:0", "no message received on ys");
+
+ start(xs);
+
+
+ notifyObservers(message(2), topic, null);
+
+ assert.deepEqual(input.value, {data:2}, "message received on input port");
+ assert.deepEqual(xs.value, "x:2", "message received on xs");
+ assert.deepEqual(ys.value, "y:2", "no message received on (not started) ys");
+
+
+ notifyObservers(message(3), topic, null);
+
+
+ assert.deepEqual(input.value, {data:3}, "message received on input port");
+ assert.deepEqual(xs.value, "x:3", "message received on xs");
+ assert.deepEqual(ys.value, "y:3", "message received on ys");
+
+
+ notifyObservers(message(4), topic, null);
+
+ assert.deepEqual(input.value, {data:4}, "message received on input port");
+ assert.deepEqual(xs.value, "x:4", "message not received on (stopped) xs");
+ assert.deepEqual(ys.value, "y:4", "message received on ys");
+
+
+ stop(input);
+
+ notifyObservers(message(5), topic, null);
+
+ assert.deepEqual(input.value, {data:4}, "message note received on input port");
+ assert.deepEqual(xs.value, "x:4", "message not received on (stopped) xs");
+ assert.deepEqual(ys.value, "y:4", "message not received on (stopped) ys");
+};
+
+exports["test send messages to nsIObserverService"] = assert => {
+ let messages = [];
+
+ const { newURI } = Cc['@mozilla.org/network/io-service;1'].
+ getService(Ci.nsIIOService);
+
+ const output = new OutputPort({ id: Date.now().toString(32), sync: true });
+ const topic = output.topic;
+
+ const observer = {
+ QueryInterface: function() {
+ return this;
+ },
+ observe: (subject, topic, data) => {
+ // Ignores internal console events
+ if (!isConsoleEvent(topic)) {
+ messages.push({
+ topic: topic,
+ subject: subject
+ });
+ }
+ }
+ };
+
+ addObserver(observer, topic, false);
+
+ send(output, null);
+ assert.deepEqual(messages.shift(), { topic: topic, subject: null },
+ "null message received");
+
+
+ const uri = newURI("http://www.foo.com", null, null);
+ send(output, uri);
+
+ assert.deepEqual(messages.shift(), { topic: topic, subject: uri },
+ "message received");
+
+
+ function customSubject() {}
+ send(output, customSubject);
+
+ let message = messages.shift();
+ assert.equal(message.topic, topic, "topic was received");
+ assert.equal(message.subject.wrappedJSObject, customSubject,
+ "custom subject is received");
+
+ removeObserver(observer, topic);
+
+ send(output, { data: "more data" });
+
+ assert.deepEqual(messages, [],
+ "no more data received");
+
+ addObserver(observer, "*", false);
+
+ send(output, { data: "data again" });
+
+ message = messages.shift();
+ assert.equal(message.topic, topic, "topic was received");
+ assert.deepEqual(message.subject.wrappedJSObject,
+ { data: "data again" },
+ "wrapped message received");
+
+ removeObserver(observer, "*");
+
+ send(output, { data: "last data" });
+ assert.deepEqual(messages, [],
+ "no more data received");
+
+ assert.throws(() => send(output, "hi"),
+ /Unsupproted message type: `string`/,
+ "strings can't be send");
+
+ assert.throws(() => send(output, 4),
+ /Unsupproted message type: `number`/,
+ "numbers can't be send");
+
+ assert.throws(() => send(output, void(0)),
+ /Unsupproted message type: `undefined`/,
+ "undefineds can't be send");
+
+ assert.throws(() => send(output, true),
+ /Unsupproted message type: `boolean`/,
+ "booleans can't be send");
+};
+
+exports["test async OutputPort"] = (assert, done) => {
+ let async = false;
+ const output = new OutputPort({ id: Date.now().toString(32) });
+ const observer = {
+ observe: (subject, topic, data) => {
+ removeObserver(observer, topic);
+ assert.equal(topic, output.topic, "correct topic");
+ assert.deepEqual(subject.wrappedJSObject, {foo: "bar"}, "message received");
+ assert.ok(async, "message received async");
+ done();
+ }
+ };
+ addObserver(observer, output.topic, false);
+ send(output, {foo: "bar"});
+
+ assert.throws(() => send(output, "boom"), "can only send object");
+ async = true;
+};
+
+exports["test explicit output topic"] = (assert, done) => {
+ const topic = Date.now().toString(32);
+ const output = new OutputPort({ topic: topic });
+ const observer = {
+ observe: (subject, topic, data) => {
+ removeObserver(observer, topic);
+ assert.deepEqual(subject.wrappedJSObject, {foo: "bar"}, "message received");
+ done();
+ }
+ };
+
+ assert.equal(output.topic, topic, "given topic is used");
+
+ addObserver(observer, topic, false);
+ send(output, {foo: "bar"});
+};
+
+exports["test explicit input topic"] = (assert) => {
+ const topic = Date.now().toString(32);
+ const input = new InputPort({ topic: topic });
+
+ start(input);
+ assert.equal(input.topic, topic, "given topic is used");
+
+
+ notifyObservers({wrappedJSObject: {foo: "bar"}}, topic, null);
+
+ assert.deepEqual(input.value, {foo: "bar"}, "message received");
+};
+
+
+exports["test receive what was send"] = assert => {
+ const id = Date.now().toString(32);
+ const input = new InputPort({ id: id, initial: 0});
+ const output = new OutputPort({ id: id, sync: true });
+
+ assert.ok(input.topic.includes(addonID),
+ "input topic is namespaced to addon");
+ assert.equal(input.topic, output.topic,
+ "input & output get same topics from id.");
+
+ start(input);
+
+ assert.equal(input.value, 0, "initial value is set");
+
+ send(output, { data: 1 });
+ assert.deepEqual(input.value, {data: 1}, "message unwrapped");
+
+ send(output, []);
+ assert.deepEqual(input.value, [], "array message unwrapped");
+
+ send(output, null);
+ assert.deepEqual(input.value, null, "null message received");
+
+ send(output, new String("message"));
+ assert.deepEqual(input.value, new String("message"),
+ "string instance received");
+
+ send(output, /pattern/);
+ assert.deepEqual(input.value, /pattern/, "regexp received");
+
+ assert.throws(() => send(output, "hi"),
+ /Unsupproted message type: `string`/,
+ "strings can't be send");
+
+ assert.throws(() => send(output, 4),
+ /Unsupproted message type: `number`/,
+ "numbers can't be send");
+
+ assert.throws(() => send(output, void(0)),
+ /Unsupproted message type: `undefined`/,
+ "undefineds can't be send");
+
+ assert.throws(() => send(output, true),
+ /Unsupproted message type: `boolean`/,
+ "booleans can't be send");
+
+ stop(input);
+};
+
+
+exports["-test error reporting"] = function(assert) {
+ let { loader, messages } = LoaderWithHookedConsole2(module);
+ const { start, stop, lift } = loader.require("sdk/event/utils");
+ const { InputPort } = loader.require("sdk/input/system");
+ const { OutputPort } = loader.require("sdk/output/system");
+ const id = "error:" + Date.now().toString(32);
+
+ const raise = x => { if (x) throw new Error("foo"); };
+
+ const input = new InputPort({ id: id });
+ const output = new OutputPort({ id: id, sync: true });
+ const xs = lift(raise, input);
+
+ assert.equal(input.value, null, "initial inherited");
+
+ send(output, { data: "yo yo" });
+
+ assert.deepEqual(messages, [], "nothing happend yet");
+
+ start(xs);
+
+ send(output, { data: "first" });
+
+ assert.equal(messages.length, 4, "Got an exception");
+
+
+ assert.equal(messages[0], "console.error: " + addonName + ": \n",
+ "error is logged");
+
+ assert.ok(/Unhandled error/.test(messages[1]),
+ "expected error message");
+
+ loader.unload();
+};
+
+exports["test unload ends input port"] = assert => {
+ const loader = Loader(module);
+ const { start, stop, lift } = loader.require("sdk/event/utils");
+ const { InputPort } = loader.require("sdk/input/system");
+
+ const id = "unload!" + Date.now().toString(32);
+ const input = new InputPort({ id: id });
+
+ start(input);
+ notifyObservers(message(1), input.topic, null);
+ assert.deepEqual(input.value, {data: 1}, "message received");
+
+ notifyObservers(message(2), input.topic, null);
+ assert.deepEqual(input.value, {data: 2}, "message received");
+
+ loader.unload();
+ notifyObservers(message(3), input.topic, null);
+ assert.deepEqual(input.value, {data: 2}, "message wasn't received");
+};
+
+require("sdk/test").run(exports);