1 /******************************************************************************
2 * Copyright 2012 Intel Corporation.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 *****************************************************************************/
19 /*****************************************************************************/
21 var dbus = { // hook object for dbus types not translated by python-json
22 Double: function(value, level) {
29 /*****************************************************************************/
31 var cloudeebus = window.cloudeebus = {
36 cloudeebus.reset = function() {
37 cloudeebus.sessionBus = null;
38 cloudeebus.systemBus = null;
39 cloudeebus.wampSession = null;
40 cloudeebus.uri = null;
44 cloudeebus.log = function(msg) {
48 cloudeebus.versionCheck = function(version) {
49 var ver = version.split(".");
50 var min = cloudeebus.minVersion.split(".");
51 for (var i=0; i<ver.length; i++) {
52 if (Number(ver[i]) > Number(min[i]))
54 if (Number(ver[i]) < Number(min[i]))
61 cloudeebus.connect = function(uri, manifest, successCB, errorCB) {
65 function onCloudeebusVersionCheckCB(version) {
66 if (cloudeebus.versionCheck(version)) {
67 cloudeebus.log("Connected to " + cloudeebus.uri);
71 var errorMsg = "Cloudeebus server version " + version + " unsupported, need version " + cloudeebus.minVersion + " or superior";
72 cloudeebus.log(errorMsg);
78 function onWAMPSessionAuthErrorCB(error) {
79 cloudeebus.log("Authentication error: " + error.desc);
84 function onWAMPSessionAuthenticatedCB(permissions) {
85 cloudeebus.sessionBus = new cloudeebus.BusConnection("session", cloudeebus.wampSession);
86 cloudeebus.systemBus = new cloudeebus.BusConnection("system", cloudeebus.wampSession);
87 cloudeebus.wampSession.call("getVersion").then(onCloudeebusVersionCheckCB, errorCB);
90 function onWAMPSessionChallengedCB(challenge) {
91 var signature = cloudeebus.wampSession.authsign(challenge, manifest.key);
92 cloudeebus.wampSession.auth(signature).then(onWAMPSessionAuthenticatedCB, onWAMPSessionAuthErrorCB);
95 function onWAMPSessionConnectedCB(session) {
96 cloudeebus.wampSession = session;
98 cloudeebus.wampSession.authreq(
100 {permissions: manifest.permissions}
101 ).then(onWAMPSessionChallengedCB, onWAMPSessionAuthErrorCB);
103 cloudeebus.wampSession.authreq().then(function() {
104 cloudeebus.wampSession.auth().then(onWAMPSessionAuthenticatedCB, onWAMPSessionAuthErrorCB);
105 }, onWAMPSessionAuthErrorCB);
108 function onWAMPSessionErrorCB(code, reason) {
109 if (code == ab.CONNECTION_UNSUPPORTED) {
110 cloudeebus.log("Browser is not supported");
113 cloudeebus.log("Failed to open session, code = " + code + ", reason = " + reason);
119 return ab.connect(cloudeebus.uri, onWAMPSessionConnectedCB, onWAMPSessionErrorCB);
123 cloudeebus.SessionBus = function() {
124 return cloudeebus.sessionBus;
128 cloudeebus.SystemBus = function() {
129 return cloudeebus.systemBus;
134 /*****************************************************************************/
136 cloudeebus.BusConnection = function(name, session) {
138 this.wampSession = session;
143 cloudeebus.BusConnection.prototype.getObject = function(busName, objectPath, introspectCB, errorCB) {
144 var proxy = new cloudeebus.ProxyObject(this.wampSession, this, busName, objectPath);
146 proxy._introspect(introspectCB, errorCB);
152 /*****************************************************************************/
154 function _processWrappers(wrappers, value) {
155 for (var i=0; i<wrappers.length; i++)
160 function _processWrappersAsync(wrappers, value) {
162 function processAsyncOnce() {
163 _processWrappers(wrappers, value);
164 clearInterval(taskid);
166 taskid = setInterval(processAsyncOnce, 200);
171 /*****************************************************************************/
173 cloudeebus.FutureResolver = function(future) {
174 this.future = future;
175 this.resolved = null;
180 cloudeebus.FutureResolver.prototype.resolve = function(value, sync) {
184 var then = (value && value.then && value.then.apply) ? value.then : null;
187 var acceptCallback = function(arg) {
188 self.resolve(arg, true);
190 var rejectCallback = function(arg) {
191 self.reject(arg, true);
194 then.apply(value, [acceptCallback, rejectCallback]);
197 this.reject(e, true);
201 this.accept(value, sync);
205 cloudeebus.FutureResolver.prototype.accept = function(value, sync) {
209 var future = this.future;
210 future.state = "accepted";
211 future.result = value;
213 this.resolved = true;
215 _processWrappers(future._acceptWrappers, value);
217 _processWrappersAsync(future._acceptWrappers, value);
221 cloudeebus.FutureResolver.prototype.reject = function(value, sync) {
225 var future = this.future;
226 future.state = "rejected";
227 future.result = value;
229 this.resolved = true;
231 _processWrappers(future._rejectWrappers, value);
233 _processWrappersAsync(future._rejectWrappers, value);
238 /*****************************************************************************/
240 cloudeebus.Future = function(init) {
241 this.state = "pending";
243 this._acceptWrappers = [];
244 this._rejectWrappers = [];
245 this.resolver = new cloudeebus.FutureResolver(this);
248 init.apply(this, [this.resolver]);
251 this.resolver.reject(e, true);
258 cloudeebus.Future.prototype.appendWrappers = function(acceptWrapper, rejectWrapper) {
260 this._acceptWrappers.push(acceptWrapper);
262 this._rejectWrappers.push(rejectWrapper);
263 if (this.state == "accepted")
264 _processWrappersAsync(this._acceptWrappers, this.result);
265 if (this.state == "rejected")
266 _processWrappersAsync(this._rejectWrappers, this.result);
270 cloudeebus.Future.prototype.then = function(acceptCB, rejectCB) {
271 var future = new cloudeebus.Future();
272 var resolver = future.resolver;
273 var acceptWrapper, rejectWrapper;
276 acceptWrapper = function(arg) {
278 var value = acceptCB.apply(future, [arg]);
279 resolver.resolve(value, true);
282 resolver.reject(e, true);
286 acceptWrapper = function(arg) {
287 resolver.accept(arg, true);
291 rejectWrapper = function(arg) {
293 var value = rejectCB.apply(future, [arg]);
294 resolver.resolve(value, true);
297 resolver.reject(e, true);
301 rejectWrapper = function(arg) {
302 resolver.reject(arg, true);
305 this.appendWrappers(acceptWrapper,rejectWrapper);
310 cloudeebus.Future.prototype["catch"] = function(rejectCB) {
311 return this.then(undefined,rejectCB);
315 cloudeebus.Future.prototype.done = function(acceptCB, rejectCB) {
316 this.appendWrappers(acceptCB,rejectCB);
320 cloudeebus.Future.resolve = function(value) {
321 var future = new cloudeebus.Future();
322 future.resolver.resolve(value);
327 cloudeebus.Future.accept = function(value) {
328 var future = new cloudeebus.Future();
329 future.resolver.accept(value);
334 cloudeebus.Future.reject = function(value) {
335 var future = new cloudeebus.Future();
336 future.resolver.reject(value);
341 cloudeebus.Future.any = function() {
342 var future = new cloudeebus.Future();
343 var resolver = future.resolver;
344 var acceptCallback = function(arg) {
345 resolver.resolve(arg, true);
347 var rejectCallback = function(arg) {
348 resolver.reject(arg, true);
350 if (arguments.length == 0)
351 resolver.resolve(undefined, true);
354 Future.resolve(arguments[i]).appendWrappers(acceptCallback,rejectCallback);
359 cloudeebus.Future.every = function() {
360 var future = new cloudeebus.Future();
361 var resolver = future.resolver;
363 var countdown = arguments.length;
364 var args = new Array(countdown);
365 var rejectCallback = function(arg) {
366 resolver.reject(arg, true);
368 if (arguments.length == 0)
369 resolver.resolve(undefined, true);
371 for (i in arguments) {
372 var acceptCallback = function(arg) {
376 resolver.resolve(args, true);
379 Future.resolve(arguments[i]).appendWrappers(acceptCallback,rejectCallback);
386 cloudeebus.Future.some = function() {
387 var future = new cloudeebus.Future();
388 var resolver = future.resolver;
390 var countdown = arguments.length;
391 var args = new Array(countdown);
392 var acceptCallback = function(arg) {
393 resolver.resolve(arg, true);
395 if (arguments.length == 0)
396 resolver.resolve(undefined, true);
398 for (i in arguments) {
399 var rejectCallback = function(arg) {
403 resolver.reject(args, true);
406 Future.resolve(arguments[i]).appendWrappers(acceptCallback,rejectCallback);
414 /*****************************************************************************/
416 cloudeebus.ProxyObject = function(session, busConnection, busName, objectPath) {
417 this.wampSession = session;
418 this.busConnection = busConnection;
419 this.busName = busName;
420 this.objectPath = objectPath;
421 this.interfaceProxies = {};
426 cloudeebus.ProxyObject.prototype.getInterface = function(ifName) {
427 return this.interfaceProxies[ifName];
431 cloudeebus.ProxyObject.prototype._introspect = function(successCB, errorCB) {
435 function getAllPropertiesSuccessCB(props) {
436 var ifProxy = self.interfaceProxies[self.propInterfaces[self.propInterfaces.length-1]];
437 for (var prop in props)
438 ifProxy[prop] = self[prop] = props[prop];
439 getAllPropertiesNextInterfaceCB();
442 function getAllPropertiesNextInterfaceCB() {
443 self.propInterfaces.pop();
444 if (self.propInterfaces.length > 0)
445 self.callMethod("org.freedesktop.DBus.Properties",
447 [self.propInterfaces[self.propInterfaces.length-1]]).then(getAllPropertiesSuccessCB,
448 errorCB ? errorCB : getAllPropertiesNextInterfaceCB);
450 self.propInterfaces = null;
456 function introspectSuccessCB(str) {
457 var parser = new DOMParser();
458 var xmlDoc = parser.parseFromString(str, "text/xml");
459 var interfaces = xmlDoc.getElementsByTagName("interface");
460 self.propInterfaces = [];
461 var supportDBusProperties = false;
462 for (var i=0; i < interfaces.length; i++) {
463 var ifName = interfaces[i].attributes.getNamedItem("name").value;
464 self.interfaceProxies[ifName] = new cloudeebus.ProxyObject(self.wampSession, self.busConnection, self.busName, self.objectPath);
465 if (ifName == "org.freedesktop.DBus.Properties")
466 supportDBusProperties = true;
467 var hasProperties = false;
468 var ifChild = interfaces[i].firstChild;
470 if (ifChild.nodeName == "method") {
473 var metChild = ifChild.firstChild;
475 if (metChild.nodeName == "arg" &&
476 metChild.attributes.getNamedItem("direction").value == "in") {
477 signature += metChild.attributes.getNamedItem("type").value;
480 metChild = metChild.nextSibling;
482 var metName = ifChild.attributes.getNamedItem("name").value;
484 self._addMethod(ifName, metName, nArgs, signature);
485 self.interfaceProxies[ifName]._addMethod(ifName, metName, nArgs, signature);
487 else if (ifChild.nodeName == "property") {
489 self.propInterfaces.push(ifName);
490 hasProperties = true;
492 ifChild = ifChild.nextSibling;
495 if (supportDBusProperties && self.propInterfaces.length > 0) {
496 self.callMethod("org.freedesktop.DBus.Properties",
498 [self.propInterfaces[self.propInterfaces.length-1]]).then(getAllPropertiesSuccessCB,
499 errorCB ? errorCB : getAllPropertiesNextInterfaceCB);
502 self.propInterfaces = null;
508 // call Introspect on self
509 self.callMethod("org.freedesktop.DBus.Introspectable", "Introspect", []).then(introspectSuccessCB, errorCB);
513 cloudeebus.ProxyObject.prototype._addMethod = function(ifName, method, nArgs, signature) {
517 self[method] = function() {
519 for (var i=0; i < nArgs; i++ )
520 args.push(arguments[i]);
521 return self.callMethod(ifName, method, args, signature);
526 cloudeebus.ProxyObject.prototype.callMethod = function(ifName, method, args, signature) {
530 var future = new cloudeebus.Future(function (resolver) {
531 function callMethodSuccessCB(str) {
532 try { // calling dbus hook object function for un-translated types
533 var result = eval(str);
534 resolver.accept(result[0], true);
537 cloudeebus.log("Method callback exception: " + e);
538 resolver.reject(e, true);
542 function callMethodErrorCB(error) {
543 cloudeebus.log("Error calling method: " + method + " on object: " + self.objectPath + " : " + error.desc);
544 resolver.reject(error.desc, true);
548 self.busConnection.name,
556 // call dbusSend with bus type, destination, object, message and arguments
557 self.wampSession.call("dbusSend", arglist).then(callMethodSuccessCB, callMethodErrorCB);
564 cloudeebus.ProxyObject.prototype.connectToSignal = function(ifName, signal, successCB, errorCB) {
568 function signalHandler(id, data) {
570 try { // calling dbus hook object function for un-translated types
571 successCB.apply(self, eval(data));
574 cloudeebus.log("Signal handler exception: " + e);
581 function connectToSignalSuccessCB(str) {
583 self.wampSession.subscribe(str, signalHandler);
586 cloudeebus.log("Subscribe error: " + e);
590 function connectToSignalErrorCB(error) {
591 cloudeebus.log("Error connecting to signal: " + signal + " on object: " + self.objectPath + " : " + error.desc);
597 self.busConnection.name,
604 // call dbusSend with bus type, destination, object, message and arguments
605 self.wampSession.call("dbusRegister", arglist).then(connectToSignalSuccessCB, connectToSignalErrorCB);
609 cloudeebus.ProxyObject.prototype.disconnectSignal = function(ifName, signal) {
611 this.wampSession.unsubscribe(this.busConnection.name + "#" + this.busName + "#" + this.objectPath + "#" + ifName + "#" + signal);
614 cloudeebus.log("Unsubscribe error: " + e);