forked from mirrors/gecko-dev
		
	 166d8573cb
			
		
	
	
		166d8573cb
		
	
	
	
	
		
			
			Depends on D167518 Differential Revision: https://phabricator.services.mozilla.com/D167519
		
			
				
	
	
		
			281 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			281 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* 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/. */
 | |
| 
 | |
| /*
 | |
|  * Implements low-overhead integration between components of the application.
 | |
|  * This may have different uses depending on the component, including:
 | |
|  *
 | |
|  * - Providing product-specific implementations registered at startup.
 | |
|  * - Using alternative implementations during unit tests.
 | |
|  * - Allowing add-ons to change specific behaviors.
 | |
|  *
 | |
|  * Components may define one or more integration points, each defined by a
 | |
|  * root integration object whose properties and methods are the public interface
 | |
|  * and default implementation of the integration point. For example:
 | |
|  *
 | |
|  *   const DownloadIntegration = {
 | |
|  *     getTemporaryDirectory() {
 | |
|  *       return "/tmp/";
 | |
|  *     },
 | |
|  *
 | |
|  *     getTemporaryFile(name) {
 | |
|  *       return this.getTemporaryDirectory() + name;
 | |
|  *     },
 | |
|  *   };
 | |
|  *
 | |
|  * Other parts of the application may register overrides for some or all of the
 | |
|  * defined properties and methods. The component defining the integration point
 | |
|  * does not have to be loaded at this stage, because the name of the integration
 | |
|  * point is the only information required. For example, if the integration point
 | |
|  * is called "downloads":
 | |
|  *
 | |
|  *   Integration.downloads.register(base => ({
 | |
|  *     getTemporaryDirectory() {
 | |
|  *       return base.getTemporaryDirectory.call(this) + "subdir/";
 | |
|  *     },
 | |
|  *   }));
 | |
|  *
 | |
|  * When the component defining the integration point needs to call a method on
 | |
|  * the integration object, instead of using it directly the component would use
 | |
|  * the "getCombined" method to retrieve an object that includes all overrides.
 | |
|  * For example:
 | |
|  *
 | |
|  *   let combined = Integration.downloads.getCombined(DownloadIntegration);
 | |
|  *   Assert.is(combined.getTemporaryFile("file"), "/tmp/subdir/file");
 | |
|  *
 | |
|  * Overrides can be registered at startup or at any later time, so each call to
 | |
|  * "getCombined" may return a different object. The simplest way to create a
 | |
|  * reference to the combined object that stays updated to the latest version is
 | |
|  * to define the root object in a JSM and use the "defineModuleGetter" method.
 | |
|  *
 | |
|  * *** Registration ***
 | |
|  *
 | |
|  * Since the interface is not declared formally, the registrations can happen
 | |
|  * at startup without loading the component, so they do not affect performance.
 | |
|  *
 | |
|  * Hovever, this module does not provide a startup registry, this means that the
 | |
|  * code that registers and implements the override must be loaded at startup.
 | |
|  *
 | |
|  * If performance for the override code is a concern, you can take advantage of
 | |
|  * the fact that the function used to create the override is called lazily, and
 | |
|  * include only a stub loader for the final code in an existing startup module.
 | |
|  *
 | |
|  * The registration of overrides should be repeated for each process where the
 | |
|  * relevant integration methods will be called.
 | |
|  *
 | |
|  * *** Accessing base methods and properties ***
 | |
|  *
 | |
|  * Overrides are included in the prototype chain of the combined object in the
 | |
|  * same order they were registered, where the first is closest to the root.
 | |
|  *
 | |
|  * When defining overrides, you do not need to manipulate the prototype chain of
 | |
|  * the objects you create, because their properties and methods are moved to a
 | |
|  * new object with the correct prototype. If you do, however, you can call base
 | |
|  * properties and methods using the "super" keyword. For example:
 | |
|  *
 | |
|  *   Integration.downloads.register(base => {
 | |
|  *     let newObject = {
 | |
|  *       getTemporaryDirectory() {
 | |
|  *         return super.getTemporaryDirectory() + "subdir/";
 | |
|  *       },
 | |
|  *     };
 | |
|  *     Object.setPrototypeOf(newObject, base);
 | |
|  *     return newObject;
 | |
|  *   });
 | |
|  *
 | |
|  * *** State handling ***
 | |
|  *
 | |
|  * Storing state directly on the combined integration object using the "this"
 | |
|  * reference is not recommended. When a new integration is registered, own
 | |
|  * properties stored on the old combined object are copied to the new combined
 | |
|  * object using a shallow copy, but the "this" reference for new invocations
 | |
|  * of the methods will be different.
 | |
|  *
 | |
|  * If the root object defines a property that always points to the same object,
 | |
|  * for example a "state" property, you can safely use it across registrations.
 | |
|  *
 | |
|  * Integration overrides provided by restartless add-ons should not use the
 | |
|  * "this" reference to store state, to avoid conflicts with other add-ons.
 | |
|  *
 | |
|  * *** Interaction with XPCOM ***
 | |
|  *
 | |
|  * Providing the combined object as an argument to any XPCOM method will
 | |
|  * generate a console error message, and will throw an exception where possible.
 | |
|  * For example, you cannot register observers directly on the combined object.
 | |
|  * This helps preventing mistakes due to the fact that the combined object
 | |
|  * reference changes when new integration overrides are registered.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * Maps integration point names to IntegrationPoint objects.
 | |
|  */
 | |
| const gIntegrationPoints = new Map();
 | |
| 
 | |
| /**
 | |
|  * This Proxy object creates IntegrationPoint objects using their name as key.
 | |
|  * The objects will be the same for the duration of the process. For example:
 | |
|  *
 | |
|  *   Integration.downloads.register(...);
 | |
|  *   Integration["addon-provided-integration"].register(...);
 | |
|  */
 | |
| export var Integration = new Proxy(
 | |
|   {},
 | |
|   {
 | |
|     get(target, name) {
 | |
|       let integrationPoint = gIntegrationPoints.get(name);
 | |
|       if (!integrationPoint) {
 | |
|         integrationPoint = new IntegrationPoint();
 | |
|         gIntegrationPoints.set(name, integrationPoint);
 | |
|       }
 | |
|       return integrationPoint;
 | |
|     },
 | |
|   }
 | |
| );
 | |
| 
 | |
| /**
 | |
|  * Individual integration point for which overrides can be registered.
 | |
|  */
 | |
| var IntegrationPoint = function() {
 | |
|   this._overrideFns = new Set();
 | |
|   this._combined = {
 | |
|     // eslint-disable-next-line mozilla/use-chromeutils-generateqi
 | |
|     QueryInterface() {
 | |
|       let ex = new Components.Exception(
 | |
|         "Integration objects should not be used with XPCOM because" +
 | |
|           " they change when new overrides are registered.",
 | |
|         Cr.NS_ERROR_NO_INTERFACE
 | |
|       );
 | |
|       console.error(ex);
 | |
|       throw ex;
 | |
|     },
 | |
|   };
 | |
| };
 | |
| 
 | |
| IntegrationPoint.prototype = {
 | |
|   /**
 | |
|    * Ordered set of registered functions defining integration overrides.
 | |
|    */
 | |
|   _overrideFns: null,
 | |
| 
 | |
|   /**
 | |
|    * Combined integration object. When this reference changes, properties
 | |
|    * defined directly on this object are copied to the new object.
 | |
|    *
 | |
|    * Initially, the only property of this object is a "QueryInterface" method
 | |
|    * that throws an exception, to prevent misuse as a permanent XPCOM listener.
 | |
|    */
 | |
|   _combined: null,
 | |
| 
 | |
|   /**
 | |
|    * Indicates whether the integration object is current based on the list of
 | |
|    * registered integration overrides.
 | |
|    */
 | |
|   _combinedIsCurrent: false,
 | |
| 
 | |
|   /**
 | |
|    * Registers new overrides for the integration methods. For example:
 | |
|    *
 | |
|    *   Integration.nameOfIntegrationPoint.register(base => ({
 | |
|    *     asyncMethod: Task.async(function* () {
 | |
|    *       return yield base.asyncMethod.apply(this, arguments);
 | |
|    *     }),
 | |
|    *   }));
 | |
|    *
 | |
|    * @param overrideFn
 | |
|    *        Function returning an object defining the methods that should be
 | |
|    *        overridden. Its only parameter is an object that contains the base
 | |
|    *        implementation of all the available methods.
 | |
|    *
 | |
|    * @note The override function is called every time the list of registered
 | |
|    *       override functions changes. Thus, it should not have any side
 | |
|    *       effects or do any other initialization.
 | |
|    */
 | |
|   register(overrideFn) {
 | |
|     this._overrideFns.add(overrideFn);
 | |
|     this._combinedIsCurrent = false;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Removes a previously registered integration override.
 | |
|    *
 | |
|    * Overrides don't usually need to be unregistered, unless they are added by a
 | |
|    * restartless add-on, in which case they should be unregistered when the
 | |
|    * add-on is disabled or uninstalled.
 | |
|    *
 | |
|    * @param overrideFn
 | |
|    *        This must be the same function object passed to "register".
 | |
|    */
 | |
|   unregister(overrideFn) {
 | |
|     this._overrideFns.delete(overrideFn);
 | |
|     this._combinedIsCurrent = false;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Retrieves the dynamically generated object implementing the integration
 | |
|    * methods. Platform-specific code and add-ons can override methods of this
 | |
|    * object using the "register" method.
 | |
|    */
 | |
|   getCombined(root) {
 | |
|     if (this._combinedIsCurrent) {
 | |
|       return this._combined;
 | |
|     }
 | |
| 
 | |
|     // In addition to enumerating all the registered integration overrides in
 | |
|     // order, we want to keep any state that was previously stored in the
 | |
|     // combined object using the "this" reference in integration methods.
 | |
|     let overrideFnArray = [...this._overrideFns, () => this._combined];
 | |
| 
 | |
|     let combined = root;
 | |
|     for (let overrideFn of overrideFnArray) {
 | |
|       try {
 | |
|         // Obtain a new set of methods from the next override function in the
 | |
|         // list, specifying the current combined object as the base argument.
 | |
|         let override = overrideFn(combined);
 | |
| 
 | |
|         // Retrieve a list of property descriptors from the returned object, and
 | |
|         // use them to build a new combined object whose prototype points to the
 | |
|         // previous combined object.
 | |
|         let descriptors = {};
 | |
|         for (let name of Object.getOwnPropertyNames(override)) {
 | |
|           descriptors[name] = Object.getOwnPropertyDescriptor(override, name);
 | |
|         }
 | |
|         combined = Object.create(combined, descriptors);
 | |
|       } catch (ex) {
 | |
|         // Any error will result in the current override being skipped.
 | |
|         console.error(ex);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     this._combinedIsCurrent = true;
 | |
|     return (this._combined = combined);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Defines a getter to retrieve the dynamically generated object implementing
 | |
|    * the integration methods, loading the root implementation lazily from the
 | |
|    * specified sys.mjs module. For example:
 | |
|    *
 | |
|    *   Integration.test.defineModuleGetter(this, "TestIntegration",
 | |
|    *                    "resource://testing-common/TestIntegration.sys.mjs");
 | |
|    *
 | |
|    * @param targetObject
 | |
|    *        The object on which the lazy getter will be defined.
 | |
|    * @param name
 | |
|    *        The name of the getter to define.
 | |
|    * @param moduleUrl
 | |
|    *        The URL used to obtain the module.
 | |
|    */
 | |
|   defineESModuleGetter(targetObject, name, moduleUrl) {
 | |
|     let moduleHolder = {};
 | |
|     // eslint-disable-next-line mozilla/lazy-getter-object-name
 | |
|     ChromeUtils.defineESModuleGetters(moduleHolder, {
 | |
|       [name]: moduleUrl,
 | |
|     });
 | |
|     Object.defineProperty(targetObject, name, {
 | |
|       get: () => this.getCombined(moduleHolder[name]),
 | |
|       configurable: true,
 | |
|       enumerable: true,
 | |
|     });
 | |
|   },
 | |
| };
 |