| Index: mojo/services/public/js/mojo.js
|
| diff --git a/mojo/services/public/js/mojo.js b/mojo/services/public/js/mojo.js
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..ad0ed0c3a0984be8e4c776b2605266ae22d19c67
|
| --- /dev/null
|
| +++ b/mojo/services/public/js/mojo.js
|
| @@ -0,0 +1,169 @@
|
| +// Copyright 2014 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +define("mojo/services/public/js/mojo", [
|
| + "mojo/public/interfaces/application/service_provider.mojom",
|
| + "mojo/public/js/connection",
|
| + "mojo/public/js/core",
|
| + "services/js/bridge",
|
| +], function(service, connection, core, bridge) {
|
| +
|
| + 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) {
|
| + // 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(pipe.handle0, clientClass, service.proxyClass);
|
| + if (serviceConnection.local)
|
| + serviceConnection.local.delegate$ = client;
|
| +
|
| + 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.requestor = requestor;
|
| + exports.shell = shell;
|
| + exports.quit = quit;
|
| + return exports;
|
| +});
|
|
|