Bug 238324. Tweak generateFactory to call new, and add unit tests for module loading. Patch by Alex Vincent <ajvincent@gmail.com>. r=sayrer, sr=bsmedberg

This commit is contained in:
sayrer@gmail.com 2007-05-20 15:45:14 -07:00
parent d7a24f9856
commit c1cab1cba0
3 changed files with 150 additions and 6 deletions

View File

@ -41,7 +41,7 @@
* loader.
*
* Import into a JS component using
* 'Components.utils.import("res:XPCOMUtils.js");'
* 'Components.utils.import("rel:XPCOMUtils.js");'
*
*/
@ -61,8 +61,9 @@ var XPCOMUtils = {
* not call QueryInterface on newly created objects, but instead
* check the requested iid against the interface list.
*
* @param ctor : Constructor function. A call 'ctor()' must
* return an instance of the class served by this factory.
* @param ctor : A call to 'new ctor()' must return an instance of the class
* served by this factory.
*
* @param interfaces : Optional list of interfaces. If this parameter is not
* given, objects created by 'ctor' must implement a
* QueryInterface method. If this parameter is given,
@ -74,13 +75,13 @@ var XPCOMUtils = {
createInstance: function(outer, iid) {
if (outer) throw Components.results.NS_ERROR_NO_AGGREGATION;
if (!interfaces)
return ctor().QueryInterface(iid);
return (new ctor()).QueryInterface(iid);
for (var i=interfaces.length; i>=0; --i) {
if (iid.equals(interfaces[i])) break;
}
if (i < 0 && !iid.equals(Components.interfaces.nsISupports))
throw Components.results.NS_ERROR_NO_INTERFACE;
return ctor();
return (new ctor());
}
}
},

View File

@ -0,0 +1,128 @@
/* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is Module loader test component.
*
* The Initial Developer of the Original Code is
* Alexander J. Vincent <ajvincent@gmail.com>.
* Portions created by the Initial Developer are Copyright (C) 2007
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
*
* Alternatively, the contents of this file may be used under the terms of
* either the GNU General Public License Version 2 or later (the "GPL"), or
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
* ***** END LICENSE BLOCK ***** */
function FooComponent() {
}
FooComponent.prototype =
{
// nsIClassInfo
getInterfaces: function getInterfaces(aCount) {
var interfaces = [Components.interfaces.nsIClassInfo];
aCount.value = interfaces.length;
return interfaces;
},
// nsIClassInfo
getHelperForLanguage: function getHelperForLanguage(aLanguage) {
return null;
},
// nsIClassInfo
classDescription: "Module importer test 1",
// nsIClassInfo
classID: Components.ID("{6b933fe6-6eba-4622-ac86-e4f654f1b474}"),
// nsIClassInfo
contractID: "@mozilla.org/tests/module-importer;1",
// nsIClassInfo
implementationLanguage: Components.interfaces.nsIProgrammingLanguage.JAVASCRIPT,
// nsIClassInfo
flags: 0,
// nsISupports
QueryInterface: function QueryInterface(aIID) {
if (aIID.equals(Components.interfaces.nsIClassInfo) ||
aIID.equals(Components.interfaces.nsISupports))
return this;
throw Components.results.NS_ERROR_NO_INTERFACE;
}
};
function BarComponent() {
}
BarComponent.prototype =
{
// nsIClassInfo
getInterfaces: function getInterfaces(aCount) {
var interfaces = [Components.interfaces.nsIClassInfo];
aCount.value = interfaces.length;
return interfaces;
},
// nsIClassInfo
getHelperForLanguage: function getHelperForLanguage(aLanguage) {
return null;
},
// nsIClassInfo
classDescription: "Module importer test 2",
// nsIClassInfo
classID: Components.ID("{708a896a-b48d-4bff-906e-fc2fd134b296}"),
// nsIClassInfo
contractID: "@mozilla.org/tests/module-importer;2",
// nsIClassInfo
implementationLanguage: Components.interfaces.nsIProgrammingLanguage.JAVASCRIPT,
// nsIClassInfo
flags: 0
};
Components.utils.import("rel:XPCOMUtils.jsm");
var NSGetModule = XPCOMUtils.generateNSGetModule([
{
className: FooComponent.prototype.classDescription,
cid: FooComponent.prototype.classID,
contractID: FooComponent.prototype.contractID,
factory: XPCOMUtils.generateFactory(FooComponent)
},
{
className: BarComponent.prototype.classDescription,
cid: BarComponent.prototype.classID,
contractID: BarComponent.prototype.contractID,
factory: XPCOMUtils.generateFactory(
BarComponent,
[Components.interfaces.nsIClassInfo]
)
}
], null, null);

View File

@ -21,6 +21,7 @@
*
* Contributor(s):
* Robert Sayre <sayrer@gmail.com> (original author)
* Alexander J. Vincent <ajvincent@gmail.com>
*
* Alternatively, the contents of this file may be used under the terms of
* either the GNU General Public License Version 2 or later (the "GPL"), or
@ -66,5 +67,19 @@ function run_test() {
}
do_check_true(didThrow);
}
// try to create a component
do_load_module("/js/src/xpconnect/tests/unit/component_import.js");
const contractID = "@mozilla.org/tests/module-importer;";
do_check_true((contractID + "1") in Components.classes);
var foo = Components.classes[contractID + "1"]
.createInstance(Components.interfaces.nsIClassInfo);
do_check_true(Boolean(foo));
do_check_true(foo.contractID == contractID + "1");
// try to create another component which doesn't directly implement QI
do_check_true((contractID + "2") in Components.classes);
var bar = Components.classes[contractID + "2"]
.createInstance(Components.interfaces.nsIClassInfo);
do_check_true(Boolean(bar));
do_check_true(bar.contractID == contractID + "2");
}