Index: mojo/apps/js/mojo.js |
diff --git a/mojo/apps/js/mojo.js b/mojo/apps/js/mojo.js |
index 23bacffa5290bf65f29576b6c5cdec8aa0563fc3..97021883f88a62dc6c4e30838158b92b54d48b25 100644 |
--- a/mojo/apps/js/mojo.js |
+++ b/mojo/apps/js/mojo.js |
@@ -3,24 +3,167 @@ |
// found in the LICENSE file. |
define("mojo/apps/js/mojo", [ |
+ "mojo/public/interfaces/application/service_provider.mojom", |
"mojo/public/js/bindings/connection", |
+ "mojo/public/js/bindings/core", |
"mojo/apps/js/bridge", |
-], function(connection, mojo) { |
+], function(service, connection, core, bridge) { |
- function connectToService(url, service, client) { |
- var serviceHandle = mojo.connectToService(url, service.name); |
+ function Shell() { |
+ this.applications_ = new Map(); |
+ } |
+ |
+ Shell.prototype.connectToApplication = function(url) { |
+ var application = this.applications_.get(url); |
+ if (application) |
+ return application; |
+ application = new ServiceProvider(bridge.connectToApplication(url)); |
+ this.applications_.set(url, application); |
+ return application; |
+ }; |
+ |
+ Shell.prototype.connectToService = function (url, service, client) { |
+ return this.connectToApplication(url).connectToService(service, client); |
+ }; |
+ |
+ Shell.prototype.close = function() { |
+ shell().applications_.forEach(function(application, url) { |
+ application.close(); |
+ }); |
+ shell().applications_.clear(); |
+ }; |
+ |
+ var shellValue = null; |
+ |
+ function shell() { |
+ if (!shellValue) |
+ shellValue = new Shell(); |
+ return shellValue; |
+ } |
+ |
+ var requestorValue = null; |
+ |
+ function requestor() { |
+ if (!requestorValue) { |
+ var handle = bridge.requestorMessagePipeHandle(); |
+ requestorValue = handle && new ServiceProvider(handle); |
+ } |
+ return requestorValue; |
+ } |
+ |
+ function connectToServiceImpl(serviceName, serviceHandle) { |
+ var provider = this.providers_.get(serviceName); |
+ if (!provider) { |
+ this.pendingRequests_.set(serviceName, serviceHandle); |
+ return; |
+ } |
+ |
+ var serviceConnection = new connection.Connection( |
+ serviceHandle, |
+ provider.service.delegatingStubClass, |
+ provider.service.client && provider.service.client.proxyClass); |
+ |
+ serviceConnection.local.connection$ = serviceConnection; |
+ serviceConnection.local.delegate$ = |
+ new provider.factory(serviceConnection.remote); |
+ |
+ provider.connections.push(serviceConnection); |
+ } |
+ |
+ function ServiceProvider(messagePipeHandle) { |
Aaron Boodman
2014/10/29 17:12:51
Nit: it seems like putting this in its own module
hansmuller
2014/10/29 22:43:41
Thanks for the flexibility, I will revise this in
|
+ // TODO(hansmuller): if messagePipeHandle is null, throw an exception. |
+ this.connections_ = new Map(); |
+ this.providers_ = new Map(); |
+ this.pendingRequests_ = new Map(); |
+ this.connection_ = null; |
+ this.handle_ = messagePipeHandle; |
+ this.connection_ = new connection.Connection( |
+ this.handle_, |
+ service.ServiceProvider.client.delegatingStubClass, |
+ service.ServiceProvider.proxyClass); |
+ this.connection_.local.delegate$ = { |
+ connectToService: connectToServiceImpl.bind(this) |
+ }; |
+ } |
+ |
+ ServiceProvider.prototype.provideService = function(service, factory) { |
+ // TODO(hansmuller): if !factory, remove provider and close its connections. |
+ // TODO(hansmuller): if this.connection_ is null, throw an error. |
+ var provider = { |
+ service: service, |
+ factory: factory, |
+ connections: [], |
+ }; |
+ this.providers_.set(service.name, provider); |
+ |
+ if (this.pendingRequests_.has(service.name)) { |
+ connectToServiceImpl(service.name, pendingRequests_.get(service.name)); |
+ pendingRequests_.delete(service.name); |
+ } |
+ |
+ return this; |
+ }; |
+ |
+ ServiceProvider.prototype.connectToService = function(service, client) { |
+ // TODO(hansmuler): if service.name isn't defined, throw an error. |
+ // TODO(hansmuller): if this.connection_ is null, throw an error. |
+ var serviceConnection = this.connections_.get(service.name); |
+ if (serviceConnection) |
+ return serviceConnection.remote; |
+ |
+ var pipe = core.createMessagePipe(); |
+ this.connection_.remote.connectToService(service.name, pipe.handle1); |
var clientClass = client && service.client.delegatingStubClass; |
var serviceConnection = |
- new connection.Connection(serviceHandle, clientClass, service.proxyClass); |
+ new connection.Connection(pipe.handle0, clientClass, service.proxyClass); |
if (serviceConnection.local) |
serviceConnection.local.delegate$ = client; |
- serviceConnection.remote.connection$ = serviceConnection; |
+ |
+ this.connections_.set(service.name, serviceConnection); |
return serviceConnection.remote; |
+ }; |
+ |
+ ServiceProvider.prototype.close = function() { |
+ if (!this.connection_) |
+ return; |
+ |
+ try { |
+ // Outgoing connections |
+ this.connections_.forEach(function(connection, serviceName) { |
+ connection.close(); |
+ }); |
+ // Incoming connections |
+ this.providers_.forEach(function(provider, serviceName) { |
+ provider.connections.forEach(function(connection) { |
+ connection.close(); |
+ }); |
+ }); |
+ this.connection_.close(); |
+ } finally { |
+ this.connections_ = null; |
+ this.providers_ = null; |
+ this.pendingRequests_ = null; |
+ this.connection_ = null; |
+ this.handle_ = null; |
+ |
+ shell().applications_.forEach(function(application, url) { |
+ if (application === this) |
+ shell().applications_.delete(url); |
+ }, this); |
+ } |
+ }; |
+ |
+ function quit() { |
+ if (requestorValue) |
+ requestor().close(); |
+ if (shellValue) |
+ shell().close(); |
+ bridge.quit(); |
} |
var exports = {}; |
- exports.connectToService = connectToService; |
- exports.quit = mojo.quit; |
+ exports.requestor = requestor; |
+ exports.shell = shell; |
+ exports.quit = quit; |
return exports; |
}); |
- |