home *** CD-ROM | disk | FTP | other *** search
/ PC World 2005 May / PCWorld_2005-05_cd.bin / software / vyzkuste / nvu / nvu-0.90-csCZ-installer.exe / components / nsExtensionManager.js < prev    next >
Text File  |  2005-03-10  |  177KB  |  4,533 lines

  1. /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
  2. /* ***** BEGIN LICENSE BLOCK *****
  3.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  4.  *
  5.  * The contents of this file are subject to the Mozilla Public License Version
  6.  * 1.1 (the "License"); you may not use this file except in compliance with
  7.  * the License. You may obtain a copy of the License at
  8.  * http://www.mozilla.org/MPL/
  9.  *
  10.  * Software distributed under the License is distributed on an "AS IS" basis,
  11.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  12.  * for the specific language governing rights and limitations under the
  13.  * License.
  14.  *
  15.  * The Original Code is the Extension Manager.
  16.  *
  17.  * The Initial Developer of the Original Code is Ben Goodger.
  18.  * Portions created by the Initial Developer are Copyright (C) 2004
  19.  * the Initial Developer. All Rights Reserved.
  20.  *
  21.  * Contributor(s):
  22.  *  Ben Goodger <ben@bengoodger.com>
  23.  *
  24.  * Alternatively, the contents of this file may be used under the terms of
  25.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  26.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  27.  * in which case the provisions of the GPL or the LGPL are applicable instead
  28.  * of those above. If you wish to allow use of your version of this file only
  29.  * under the terms of either the GPL or the LGPL, and not to allow others to
  30.  * use your version of this file under the terms of the MPL, indicate your
  31.  * decision by deleting the provisions above and replace them with the notice
  32.  * and other provisions required by the GPL or the LGPL. If you do not delete
  33.  * the provisions above, a recipient may use your version of this file under
  34.  * the terms of any one of the MPL, the GPL or the LGPL.
  35.  *
  36.  * ***** END LICENSE BLOCK ***** */
  37.  
  38. const nsIExtensionManager             = Components.interfaces.nsIExtensionManager;
  39. const nsIUpdateService                = Components.interfaces.nsIUpdateService;
  40. const nsIUpdateItem                   = Components.interfaces.nsIUpdateItem;
  41.  
  42. const PREF_EM_APP_ID                  = "app.id";
  43. const PREF_EM_APP_VERSION             = "app.version";
  44. const PREF_EM_APP_EXTENSIONS_VERSION  = "app.extensions.version";
  45. const PREF_EM_APP_BUILDID             = "app.build_id";
  46. const PREF_EM_LAST_APP_VERSION        = "extensions.lastAppVersion";
  47. const PREF_UPDATE_COUNT               = "extensions.update.count";
  48. const PREF_UPDATE_DEFAULT_URL         = "extensions.update.url";
  49. const PREF_EM_WASINSAFEMODE           = "extensions.wasInSafeMode";
  50. const PREF_EM_DISABLEDOBSOLETE        = "extensions.disabledObsolete";
  51. const PREF_EM_LAST_SELECTED_SKIN      = "extensions.lastSelectedSkin";
  52. const PREF_EM_EXTENSION_FORMAT        = "extensions.%UUID%.";
  53. const PREF_EM_ITEM_UPDATE_ENABLED     = "extensions.%UUID%.update.enabled";
  54. const PREF_EM_ITEM_UPDATE_URL         = "extensions.%UUID%.update.url";
  55. const PREF_EM_DSS_ENABLED             = "extensions.dss.enabled";
  56.  
  57. const PREF_GENERAL_SKINS_SELECTEDSKIN = "general.skins.selectedSkin";
  58.  
  59. const DIR_EXTENSIONS                  = "extensions";
  60. const DIR_UNINSTALL                   = "uninstall";
  61. const DIR_TEMP                        = "temp";
  62. const DIR_CHROME                      = "chrome";
  63. const DIR_COMPONENTS                  = "components";
  64. const DIR_DEFAULTS                    = "defaults";
  65. const DIR_DEFAULTS_PREFS              = "preferences";
  66. const DIR_DEFAULTS_EXTENSIONS         = "extensions"; 
  67. const DIR_CR_CHROME                   = "chrome";
  68. const DIR_CR_OVERLAYINFO              = "overlayinfo";
  69. const FILE_CR_CHROMEDS                = "chrome.rdf";
  70. const FILE_EXTENSIONS                 = "Extensions.rdf";
  71. const FILE_UNINSTALL_LOG              = "Uninstall";
  72. const FILE_DEFAULTS                   = "defaults.ini";
  73. const FILE_COMPONENT_MANIFEST         = "components.ini";
  74. const FILE_COMPAT_MANIFEST            = "compatibility.ini";
  75. const FILE_INSTALL_MANIFEST           = "install.rdf";
  76. const FILE_CHROME_MANIFEST            = "contents.rdf";
  77. const FILE_WASINSAFEMODE              = "Safe Mode";
  78. const FILE_INSTALLED_EXTENSIONS       = "installed-extensions.txt"
  79. const FILE_INSTALLED_EXTENSIONS_PROCESSED = "installed-extensions-processed.txt"
  80.  
  81. const KEY_PROFILEDIR                  = "ProfD";
  82. const KEY_APPDIR                      = "XCurProcD";
  83. const KEY_DEFAULTS                    = "ProfDefNoLoc";
  84. const KEY_DEFAULT_THEME               = "classic/1.0";
  85.  
  86. const ERROR_INVALID_VERSION           = -1;
  87. const ERROR_PHONED_HOME               = -2;
  88. const ERROR_EXTENSION_IS_THEME        = -3;
  89.  
  90. var gPref           = null;
  91. var gRDF            = null;
  92. var gOS             = null;
  93. var gVersionChecker = null;
  94.  
  95. function getVersionChecker()
  96. {
  97.   if (!gVersionChecker) {
  98.     gVersionChecker = Components.classes["@mozilla.org/updates/version-checker;1"]
  99.                                 .getService(Components.interfaces.nsIVersionChecker);
  100.   }
  101.   return gVersionChecker;
  102. }
  103.  
  104. ///////////////////////////////////////////////////////////////////////////////
  105. //
  106. // Utility Functions
  107. //
  108. const EM_NS_PREFIX      = "http://www.mozilla.org/2004/em-rdf#";
  109. const CHROME_NS_PREFIX  = "http://www.mozilla.org/rdf/chrome#";
  110.  
  111. function EM_NS(aProperty)
  112. {
  113.   return EM_NS_PREFIX + aProperty;
  114. }
  115.  
  116. function CHROME_NS(aProperty)
  117. {
  118.   return CHROME_NS_PREFIX + aProperty;
  119. }
  120.  
  121. // Returns the specified directory hierarchy under the special directory 
  122. // specified by aKey, creating directories along the way if necessary.
  123. function getDir(aKey, aSubDirs)
  124. {
  125.   return getDirInternal(aKey, aSubDirs, true);
  126. }
  127.  
  128. function getDirNoCreate(aKey, aSubDirs)
  129. {
  130.   return getDirInternal(aKey, aSubDirs, false);
  131. }
  132.  
  133. function getDirInternal(aKey, aSubDirs, aCreate)
  134. {
  135.   var fileLocator = Components.classes["@mozilla.org/file/directory_service;1"]
  136.                               .getService(Components.interfaces.nsIProperties);
  137.   var dir = fileLocator.get(aKey, Components.interfaces.nsIFile);
  138.   for (var i = 0; i < aSubDirs.length; ++i) {
  139.     dir.append(aSubDirs[i]);
  140.     if (aCreate && !dir.exists())
  141.       dir.create(Components.interfaces.nsIFile.DIRECTORY_TYPE, 0755);
  142.   }
  143.   return dir;
  144. }
  145.  
  146. // Returns the file at the appropriate point in a directory hierarchy under
  147. // the specified key, creating directories along the way if necessary. Does
  148. // NOT create the file.
  149. function getFile(aKey, aPathToFile)
  150. {
  151.   var subdirs = [];
  152.   for (var i = 0; i < aPathToFile.length - 1; ++i)
  153.     subdirs.push(aPathToFile[i]);
  154.   var file = getDir(aKey, subdirs);
  155.   file.append(aPathToFile[aPathToFile.length - 1]);
  156.   return file;
  157. }
  158.  
  159. function getDirKey(aIsProfile)
  160. {
  161.   return aIsProfile ? KEY_PROFILEDIR : KEY_APPDIR;
  162. }
  163.  
  164. function dumpFile(aFile)
  165. {
  166.   dump("*** file = " + aFile.path + ", exists = " + aFile.exists() + "\n");
  167. }
  168.  
  169. // We use this to force RDF to bypass the cache when loading certain types
  170. // of files. 
  171. function getRandomFileName(aName, aExtension)
  172. {
  173.   var characters = "abcdefghijklmnopqrstuvwxyz0123456789";
  174.   var nameString = aName + "-";
  175.   for (var i = 0; i < 3; ++i) {
  176.     var index = Math.round((Math.random()) * characters.length);
  177.     nameString += characters.charAt(index);
  178.   }
  179.   return nameString + "." + aExtension;
  180. }
  181.  
  182. const PREFIX_EXTENSION  = "urn:mozilla:extension:";
  183. const PREFIX_THEME      = "urn:mozilla:theme:";
  184. const ROOT_EXTENSION    = "urn:mozilla:extension:root";
  185. const ROOT_THEME        = "urn:mozilla:theme:root";
  186.  
  187. function getItemPrefix(aItemType)
  188. {
  189.   var prefix = "";
  190.   if (aItemType & nsIUpdateItem.TYPE_EXTENSION) 
  191.     prefix = PREFIX_EXTENSION;
  192.   else if (aItemType & nsIUpdateItem.TYPE_THEME)
  193.     prefix = PREFIX_THEME;
  194.   return prefix;
  195. }
  196.  
  197. function getItemRoot(aItemType)
  198. {
  199.   var root = "";
  200.   if (aItemType & nsIUpdateItem.TYPE_EXTENSION) 
  201.     root = ROOT_EXTENSION;
  202.   else if (aItemType & nsIUpdateItem.TYPE_THEME)
  203.     root = ROOT_THEME;
  204.   return root;
  205. }
  206.  
  207. function getItemRoots(aItemType)
  208. {    
  209.   var roots = [];
  210.   if (aItemType == nsIUpdateItem.TYPE_ADDON)
  211.     roots = roots.concat([getItemRoot(nsIUpdateItem.TYPE_EXTENSION), 
  212.                           getItemRoot(nsIUpdateItem.TYPE_THEME)]);
  213.   else
  214.     roots.push(getItemRoot(aItemType));
  215.   return roots;
  216. }
  217.  
  218. function getItemType(aURI)
  219. {
  220.   var type = -1;
  221.   if (aURI.substr(0, PREFIX_EXTENSION.length) == PREFIX_EXTENSION)
  222.     type = nsIUpdateItem.TYPE_EXTENSION;
  223.   else if (aURI.substr(0, PREFIX_THEME.length) == PREFIX_THEME)
  224.     type = nsIUpdateItem.TYPE_THEME;
  225.   return type;
  226. }
  227.  
  228. function stripPrefix(aURI, aItemType)
  229. {
  230.   var val = aURI;
  231.   if (aItemType == nsIUpdateItem.TYPE_ADDON)
  232.     val = stripPrefix(aURI, getItemType(aURI));
  233.   else {
  234.     var prefix = getItemPrefix(aItemType);
  235.     if (prefix && aURI.substr(0, prefix.length) == prefix)  
  236.       val = aURI.substr(prefix.length, aURI.length);
  237.   }
  238.   return val;
  239. }
  240.  
  241. function stripPropertyPrefix(aProperty, aPrefix)
  242. {
  243.   return aProperty.substr(aPrefix.length, aProperty.length);
  244. }
  245.  
  246. function getURLSpecFromFile(aFile)
  247. {
  248.   var ioServ = Components.classes["@mozilla.org/network/io-service;1"]
  249.                           .getService(Components.interfaces.nsIIOService);
  250.   var fph = ioServ.getProtocolHandler("file").QueryInterface(Components.interfaces.nsIFileProtocolHandler);
  251.   return fph.getURLSpecFromFile(aFile);
  252. }
  253.  
  254. function ensureExtensionsFiles(aIsProfile)
  255. {
  256.   try {
  257.     var extensionsFile  = getFile(getDirKey(aIsProfile), 
  258.                                   [DIR_EXTENSIONS, FILE_EXTENSIONS]);
  259.   
  260.     // If the file does not exist at the current location, copy the default
  261.     // version over so we can access the various roots. 
  262.     // This is a sign also that something may have gone wrong, such as the user
  263.     // deleting /Extensions so we should remove the relative contents.rdf and
  264.     // overlayinfo hierarchies too. 
  265.     if (extensionsFile && !extensionsFile.exists()) {
  266.       var defaultFile = getFile(KEY_DEFAULTS, 
  267.                                 [DIR_DEFAULTS_EXTENSIONS, FILE_EXTENSIONS]);
  268.       defaultFile.copyTo(extensionsFile.parent, extensionsFile.leafName);
  269.  
  270.       // XXXben - do this only for profile until we have a better protection 
  271.       // mechanism for global items.
  272.       if (aIsProfile) {
  273.         try {      
  274.           var chromedsFile = getFile(getDirKey(aIsProfile), [DIR_CR_CHROME, FILE_CR_CHROMEDS]);
  275.           if (chromedsFile.exists())
  276.             chromedsFile.remove(false);
  277.           var overlayinfoDir = getDir(getDirKey(aIsProfile), [DIR_CR_CHROME, DIR_CR_OVERLAYINFO]);
  278.           if (overlayinfoDir.exists())
  279.             overlayinfoDir.remove(true);
  280.         }
  281.         catch (e) { 
  282.           dump("Extension System Warning: failed to remove chrome.rdf/overlay info because: " + e + "\n"); 
  283.         }
  284.       }
  285.     }
  286.   }
  287.   catch (e) { 
  288.     // Too early in the startup process to use the console, we may yet restart
  289.     // the app.
  290.     dump("Extension System Warning: Failed to set up default extensions" + 
  291.          " files probably because you do not have write privileges to this" + 
  292.          " location. While you can run Firefox like this, it is recommended" + 
  293.          " that you run it at least once with privileges that allow it to generate" + 
  294.          " these initial files to improve start performance. Running from a disk" + 
  295.          " image on MacOS X is not recommended.");
  296.   }
  297. }
  298.  
  299. function stringData(aLiteralOrResource)
  300. {
  301.   try {
  302.     var obj = aLiteralOrResource.QueryInterface(Components.interfaces.nsIRDFLiteral);
  303.     return obj.Value;
  304.   }
  305.   catch (e) {
  306.     try {
  307.       obj = aLiteralOrResource.QueryInterface(Components.interfaces.nsIRDFResource);
  308.       return obj.Value;
  309.     }
  310.     catch (e) {}
  311.   }
  312.   return "--";
  313. }
  314.  
  315. function stackTraceFunctionFormat(aFunctionName)
  316. {
  317.   var classDelimiter = aFunctionName.indexOf("_");
  318.   var className = aFunctionName.substr(0, classDelimiter);
  319.   if (!className)
  320.     className == "<global>";
  321.   var functionName = aFunctionName.substr(classDelimiter + 1, aFunctionName.length);
  322.   if (!functionName) 
  323.     functionName == "<anonymous>";
  324.   return className + "::" + functionName;
  325. }
  326.  
  327. function stackTrace(aArguments, aMaxCount)
  328. {
  329.   dump("=[STACKTRACE]=====================================================\n");
  330.   dump("*** at: " + stackTraceFunctionFormat(aArguments.callee.name) + "()\n");
  331.   var temp = aArguments.callee.caller;
  332.   var count = 0;
  333.   while (temp) {
  334.     dump("***     " + stackTraceFunctionFormat(temp.name) + "()\n");
  335.     
  336.     temp = temp.arguments.callee.caller;
  337.     if (aMaxCount > 0 && ++count == aMaxCount)
  338.       break;
  339.   }
  340.   dump("==================================================================\n");
  341. }
  342.  
  343. ///////////////////////////////////////////////////////////////////////////////
  344. // Incompatible Item Error Message
  345. function showIncompatibleError(aDS)
  346. {
  347.   var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  348.                       .getService(Components.interfaces.nsIStringBundleService);
  349.   var extensionStrings = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  350.   var params = [extensionStrings.GetStringFromName("extension")];
  351.   var title = extensionStrings.formatStringFromName("incompatibleTitle", 
  352.                                                     params, params.length);
  353.   
  354.   var brandStrings = sbs.createBundle("chrome://global/locale/brand.properties");
  355.   var brandShortName = brandStrings.GetStringFromName("brandShortName");
  356.  
  357.   var message;
  358.   var metadata = {};
  359.   getItemMetadata(aDS, metadata);
  360.   
  361.   if (undefined === metadata.minAppVersion) {
  362.     // getItemMetadata does not fill target application version range properties unless a 
  363.     // matching supported target application is found.
  364.     params = [metadata.name, metadata.version, brandShortName];
  365.     message = extensionStrings.formatStringFromName("incompatibleMessageNoApp", 
  366.                                                     params, params.length);
  367.   }
  368.   else if (metadata.minAppVersion == metadata.maxAppVersion) {
  369.     // If the min target app version and the max target app version are the same, don't show
  370.     // a message like, "Foo is only compatible with Firefox versions 0.7 to 0.7", rather just
  371.     // show, "Foo is only compatible with Firefox 0.7"
  372.     params = [metadata.name, metadata.version, brandShortName, metadata.name, 
  373.               metadata.version, brandShortName, metadata.minAppVersion];
  374.     message = extensionStrings.formatStringFromName("incompatibleMessageSingleAppVersion", 
  375.                                                     params, params.length);
  376.   }
  377.   else {
  378.     params = [metadata.name, metadata.version, brandShortName, metadata.name, 
  379.               metadata.version, brandShortName, metadata.minAppVersion, 
  380.               metadata.maxAppVersion];
  381.     message = extensionStrings.formatStringFromName("incompatibleMessage", params, params.length);
  382.   }
  383.   var ps = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
  384.                      .getService(Components.interfaces.nsIPromptService);
  385.   ps.alert(null, title, message);
  386. }
  387.  
  388. function showMalformedError(aFile)
  389. {
  390.   var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  391.                       .getService(Components.interfaces.nsIStringBundleService);
  392.   var extensionStrings = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  393.   var title = extensionStrings.GetStringFromName("malformedTitle");
  394.  
  395.   var brandStrings = sbs.createBundle("chrome://global/locale/brand.properties");
  396.   var brandShortName = brandStrings.GetStringFromName("brandShortName");
  397.   var message = extensionStrings.formatStringFromName("malformedMessage", [brandShortName, aFile], 2);
  398.   
  399.   var ps = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
  400.                      .getService(Components.interfaces.nsIPromptService);
  401.   ps.alert(null, title, message);
  402. }
  403.  
  404. function showInvalidVersionError(aItemName, aVersion)
  405. {
  406.   var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  407.                       .getService(Components.interfaces.nsIStringBundleService);
  408.   var extensionStrings = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  409.   var title = extensionStrings.GetStringFromName("invalidVersionTitle");
  410.  
  411.   var brandStrings = sbs.createBundle("chrome://global/locale/brand.properties");
  412.   var brandShortName = brandStrings.GetStringFromName("brandShortName");
  413.   var params = [brandShortName, aItemName, aVersion];
  414.   var message = extensionStrings.formatStringFromName("invalidVersionMessage", params, params.length);
  415.   
  416.   var ps = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
  417.                      .getService(Components.interfaces.nsIPromptService);
  418.   ps.alert(null, title, message);
  419. }
  420.  
  421. function showOldThemeError(aDS)
  422. {
  423.   var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  424.                       .getService(Components.interfaces.nsIStringBundleService);
  425.   var extensionStrings = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  426.   var params = [extensionStrings.GetStringFromName("theme")];
  427.   var title = extensionStrings.formatStringFromName("incompatibleTitle", 
  428.                                                     params, params.length);
  429.  
  430.   var brandStrings = sbs.createBundle("chrome://global/locale/brand.properties");
  431.   var brandShortName = brandStrings.GetStringFromName("brandShortName");
  432.   var appVersion = extensionStrings.GetStringFromName("incompatibleOlder");
  433.   
  434.   try {  
  435.     var rdfc = Components.classes["@mozilla.org/rdf/container;1"]
  436.                          .createInstance(Components.interfaces.nsIRDFContainer);
  437.     rdfc.Init(aDS, gRDF.GetResource("urn:mozilla:skin:root"));
  438.     
  439.     var elts = rdfc.GetElements();
  440.     var nameArc = gRDF.GetResource(CHROME_NS("displayName"));
  441.     while (elts.hasMoreElements()) {
  442.       var elt = elts.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  443.       themeName = stringData(aDS.GetTarget(elt, nameArc, true));
  444.       if (themeName) 
  445.         break;
  446.     }
  447.   }
  448.   catch (e) {
  449.     themeName = extensionStrings.GetStringFromName("incompatibleThemeName");
  450.   }
  451.   
  452.   params = [themeName, "", brandShortName, themeName, "", brandShortName, appVersion];
  453.   var message = extensionStrings.formatStringFromName("incompatibleMessageSingleAppVersion",
  454.                                                       params, params.length);
  455.   var ps = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
  456.                      .getService(Components.interfaces.nsIPromptService);
  457.   ps.alert(null, title, message);
  458. }
  459.  
  460. function showMissingFileError(aSourceFile, aMissingFileName)
  461. {
  462.   var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  463.                       .getService(Components.interfaces.nsIStringBundleService);
  464.   var extensionStrings = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  465.   var title = extensionStrings.GetStringFromName("missingFileTitle");
  466.  
  467.   var brandStrings = sbs.createBundle("chrome://global/locale/brand.properties");
  468.   var brandShortName = brandStrings.GetStringFromName("brandShortName");
  469.   var params = [brandShortName, aMissingFileName];
  470.   var message = extensionStrings.formatStringFromName("missingFileMessage", params, params.length);
  471.   
  472.   var ps = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
  473.                      .getService(Components.interfaces.nsIPromptService);
  474.   ps.alert(null, title, message);
  475.  
  476.   var consoleService = Components.classes["@mozilla.org/consoleservice;1"]
  477.                                  .getService(Components.interfaces.nsIConsoleService);
  478.   params = [aSourceFile, aMissingFileName];
  479.   var consoleMessage = extensionStrings.formatStringFromName("missingFileConsoleMessage",
  480.                                                              params, params.length);
  481.   consoleService.logStringMessage(consoleMessage);
  482. }
  483.  
  484. function showMalformedRegistrationError(aCRException)
  485. {
  486.   var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  487.                       .getService(Components.interfaces.nsIStringBundleService);
  488.   var extensionStrings = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  489.   var title = extensionStrings.GetStringFromName("malformedRegistrationTitle");
  490.  
  491.   var brandStrings = sbs.createBundle("chrome://global/locale/brand.properties");
  492.   var brandShortName = brandStrings.GetStringFromName("brandShortName");
  493.   var params = [brandShortName];
  494.   var message = extensionStrings.formatStringFromName("malformedRegistrationMessage", 
  495.                                                       params, params.length);
  496.   var detailsButtonMessage = extensionStrings.GetStringFromName("malformedRegistrationDetailsButton");
  497.   
  498.   var ps = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
  499.                      .getService(Components.interfaces.nsIPromptService);
  500.   var flags = (ps.BUTTON_TITLE_IS_STRING * ps.BUTTON_POS_1) +  
  501.               (ps.BUTTON_TITLE_OK * ps.BUTTON_POS_0);
  502.   var result = ps.confirmEx(null, title, message, flags, null, detailsButtonMessage, null, null, { } );
  503.   if (result == 1) {  
  504.     var consoleService = Components.classes["@mozilla.org/consoleservice;1"]
  505.                                   .getService(Components.interfaces.nsIConsoleService);
  506.     params = [aCRException.extensionID, aCRException.functionName,
  507.               aCRException.chromePath, aCRException.isProfile, ];
  508.     var consoleMessage = extensionStrings.formatStringFromName("malformedRegistrationConsoleMessage",
  509.                                                                params, params.length);
  510.     consoleService.logStringMessage(consoleMessage);
  511.     
  512.     var ww = Components.classes["@mozilla.org/embedcomp/window-watcher;1"]
  513.                         .getService(Components.interfaces.nsIWindowWatcher);
  514.     ww.openWindow(null, "chrome://global/content/console.xul", 
  515.                   "", "chrome,modal,centerscreen,resizable", null);
  516.   }
  517. }
  518.  
  519. function getItemMetadata(aDS, aResult)
  520. {
  521.   var manifestRoot = gRDF.GetResource("urn:mozilla:install-manifest");
  522.  
  523.   // Extension Name and Version
  524.   var props = ["name", "version"];
  525.   for (var i = 0; i < props.length; ++i) {
  526.     var prop = gRDF.GetResource(EM_NS(props[i]));
  527.     aResult[props[i]] = stringData(aDS.GetTarget(manifestRoot, prop, true));
  528.   }
  529.   
  530.   // Target App Name and Version
  531.   var appID = gPref.getCharPref(PREF_EM_APP_ID);
  532.  
  533.   var targets = aDS.GetTargets(manifestRoot, gRDF.GetResource(EM_NS("targetApplication")), true);
  534.   var idRes = gRDF.GetResource(EM_NS("id"));
  535.   var minVersionRes = gRDF.GetResource(EM_NS("minVersion"));
  536.   var maxVersionRes = gRDF.GetResource(EM_NS("maxVersion"));
  537.   while (targets.hasMoreElements()) {
  538.     var targetApp = targets.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  539.     var id          = stringData(aDS.GetTarget(targetApp, idRes, true));
  540.     var minVersion  = stringData(aDS.GetTarget(targetApp, minVersionRes, true));
  541.     var maxVersion  = stringData(aDS.GetTarget(targetApp, maxVersionRes, true));
  542.  
  543.     if (id == appID) {
  544.       aResult.minAppVersion = minVersion;
  545.       aResult.maxAppVersion = maxVersion;
  546.       break;
  547.     }
  548.   }
  549. }
  550.  
  551. function getInstallManifest(aFile)
  552. {
  553.   var fileURL = getURLSpecFromFile(aFile);
  554.   var ds = gRDF.GetDataSourceBlocking(fileURL);
  555.   var manifestRoot = gRDF.GetResource("urn:mozilla:install-manifest");
  556.   var arcs = ds.ArcLabelsOut(manifestRoot);
  557.   if (!arcs.hasMoreElements()) {
  558.     ds = null;
  559.     var uri = Components.classes["@mozilla.org/network/standard-url;1"]
  560.                         .createInstance(Components.interfaces.nsIURI);
  561.     uri.spec = fileURL;
  562.     var url = uri.QueryInterface(Components.interfaces.nsIURL);
  563.     showMalformedError(url.fileName);
  564.   }
  565.   return ds;
  566. }
  567.  
  568. function ArrayEnumerator(aItems)
  569. {
  570.   this._index = 0;
  571.   
  572.   if (aItems) {
  573.     for (var i = 0; i < aItems.length; ++i) {    
  574.       if (!aItems[i])
  575.         aItems.splice(i, 1);      
  576.     }
  577.   }
  578.   
  579.   this._contents = aItems;
  580. }
  581.  
  582. ArrayEnumerator.prototype = {
  583.   _index: 0,
  584.   _contents: [],
  585.   
  586.   hasMoreElements: function ArrayEnumerator_hasMoreElements()
  587.   {
  588.     return this._index < this._contents.length;
  589.   },
  590.   
  591.   getNext: function ArrayEnumerator_getNext()
  592.   {
  593.     return this._contents[this._index++];      
  594.   }
  595. };
  596.   
  597. ///////////////////////////////////////////////////////////////////////////////
  598. //
  599. // nsInstallLogBase
  600. //
  601. function nsInstallLogBase()
  602. {
  603. }
  604.  
  605. nsInstallLogBase.prototype = {
  606.   CHROME_TYPE_PACKAGE   : "package",
  607.   CHROME_TYPE_SKIN      : "skin",
  608.   CHROME_TYPE_LOCALE    : "locale",
  609.  
  610.   TOKEN_ADD_FILE        : "add",
  611.   TOKEN_REGISTER_CHROME : "register",
  612.   TOKEN_PROFILE         : "profile",
  613.   TOKEN_GLOBAL          : "global",
  614.   TOKEN_SKIN            : "skin"
  615. };
  616.  
  617. ///////////////////////////////////////////////////////////////////////////////
  618. //
  619. // nsInstallLogWriter
  620. //
  621. function nsInstallLogWriter(aExtensionID, aIsProfile)
  622. {
  623.   this._isProfile = aIsProfile;
  624.   this._uninstallLog = getDir(getDirKey(aIsProfile),
  625.                               [DIR_EXTENSIONS, aExtensionID, DIR_UNINSTALL]);
  626.   this._uninstallLog.append(FILE_UNINSTALL_LOG);
  627. }
  628.  
  629. nsInstallLogWriter.prototype = {
  630.   __proto__       : nsInstallLogBase.prototype,
  631.   _uninstallLog   : null,
  632.   
  633.   open: function nsInstallLogWriter_open ()
  634.   {
  635.     this._fos = Components.classes["@mozilla.org/network/file-output-stream;1"]
  636.                           .createInstance(Components.interfaces.nsIFileOutputStream);
  637.     const MODE_WRONLY   = 0x02;
  638.     const MODE_CREATE   = 0x08;
  639.     const MODE_TRUNCATE = 0x20;
  640.     this._fos.init(this._uninstallLog, MODE_WRONLY | MODE_CREATE | MODE_TRUNCATE, 0644, 0);
  641.   },
  642.   
  643.   close: function nsInstallLogWriter_close ()
  644.   {
  645.     this._fos.close();  
  646.   },
  647.   
  648.   addFile: function nsInstallLogWriter_addFile (aFile) 
  649.   {
  650.     var line = "add\t" + aFile.persistentDescriptor + "\n";
  651.     this._fos.write(line, line.length);
  652.   },
  653.   
  654.   registerChrome: function nsInstallLogWriter_registerChrome (aProviderName, aChromeType, aIsProfile)
  655.   {
  656.     var profile = aIsProfile ? "profile" : "global";
  657.     // register\tprofile\tpackage\t<provider_name>
  658.     var line = "register\t" + profile + "\t" + aChromeType + "\t" + aProviderName + "\n";
  659.     this._fos.write(line, line.length);
  660.   },
  661.   
  662.   installSkin: function nsInstallLogWriter_installSkin (aSkinName, aIsProfile)
  663.   {
  664.     var profile = aIsProfile ? "profile" : "global";
  665.     // register\tprofile\tpackage\t<provider_name>
  666.     var line = "skin\t" + profile + "\t" + aSkinName + "\n";
  667.     this._fos.write(line, line.length);
  668.   }
  669. };
  670.  
  671. ///////////////////////////////////////////////////////////////////////////////
  672. //
  673. // nsInstallLogReader
  674. //
  675. function nsInstallLogReader(aExtensionID, aIsProfile, aListener)
  676. {
  677.   this._isProfile = aIsProfile;
  678.   this.uninstallLog = getFile(getDirKey(aIsProfile),
  679.                               [DIR_EXTENSIONS, aExtensionID, 
  680.                                DIR_UNINSTALL, FILE_UNINSTALL_LOG]);
  681.   this._listener = aListener
  682. }
  683.  
  684. nsInstallLogReader.prototype = {
  685.   __proto__       : nsInstallLogBase.prototype,
  686.   uninstallLog    : null,
  687.   _listener       : null,
  688.   
  689.   read: function nsInstallLogReader_read ()
  690.   {
  691.     if (!this.uninstallLog.exists())
  692.       return;
  693.   
  694.     var fis = Components.classes["@mozilla.org/network/file-input-stream;1"]
  695.                         .createInstance(Components.interfaces.nsIFileInputStream);
  696.     fis.init(this.uninstallLog, -1, -1, false);
  697.     var lis = fis.QueryInterface(Components.interfaces.nsILineInputStream);
  698.     var line = { value: "" };
  699.     var more = false;
  700.     var lines = [];
  701.     do {
  702.       more = lis.readLine(line);
  703.       lines.push(line.value);
  704.     }
  705.     while (more);
  706.     fis.close();
  707.  
  708.     // Now that we've closed the stream we can remove all the files, unregister
  709.     // chrome, etc. 
  710.     //
  711.     // The list of lines we pass to the uninstall handler should be in this
  712.     // order:
  713.     // 1) File additions
  714.     // 2) Chrome Package Registrations
  715.     // 3) Chrome Skin and Locale Registrations
  716.     //
  717.     // They must be in this order since skins and locales rely on packages, and
  718.     // the packages they rely on is not stored in the registration line so we
  719.     // simply "deselect" for every package installed by the extension.
  720.     var dependentLines = [];
  721.     for (var i = 0; i < lines.length; ++i) {
  722.       var parts = lines[i].split("\t");
  723.       if (parts[1] == this.TOKEN_REGISTER_CHROME && 
  724.           (parts[2] == this.CHROME_TYPE_SKIN || 
  725.            parts[2] == this.CHROME_TYPE_LOCALE)) {
  726.         dependentLines.push(lines.splice(i, 1));
  727.       }
  728.     }
  729.     lines.concat(dependentLines);
  730.     
  731.     for (var i = 0; i < lines.length; ++i)
  732.       this._parseLine(lines[i]);
  733.   },
  734.   
  735.   _parseLine: function nsInstallLogReader__parseLine(aLine)
  736.   {
  737.     var parts = aLine.split("\t");
  738.     switch (parts[0]) {
  739.     case this.TOKEN_ADD_FILE:
  740.       var prefix = this.TOKEN_ADD_FILE + "\t";
  741.       var filePD = aLine.substr(prefix.length, aLine.length);
  742.       var lf = Components.classes["@mozilla.org/file/local;1"]
  743.                          .createInstance(Components.interfaces.nsILocalFile);
  744.       try {
  745.         lf.persistentDescriptor = filePD;
  746.         this._listener.onAddFile(lf);
  747.       }
  748.       catch (e) { 
  749.         dump("*** nsInstallLogReader::_parseLine - failed to remove file " + e + "\n"); 
  750.       }
  751.       break;
  752.     case this.TOKEN_REGISTER_CHROME:
  753.       var isProfile = parts[1] == this.TOKEN_PROFILE;
  754.       try {
  755.         this._listener.onRegisterChrome(parts[3], lf, parts[2], isProfile);
  756.       } 
  757.       catch (e) {
  758.         dump("*** nsInstallLogReader::_parseLine - failed to deregister chrome\n"); 
  759.       }
  760.       break;
  761.     case this.TOKEN_SKIN:
  762.       var isProfile = parts[1] == this.TOKEN_PROFILE;
  763.       try {
  764.         this._listener.onInstallSkin(parts[2], isProfile);
  765.       } 
  766.       catch (e) {
  767.         dump("*** nsInstallLogReader::_parseLine - failed to uninstall skin\n"); 
  768.       }
  769.       break;
  770.     }
  771.   }
  772. };
  773.  
  774. ///////////////////////////////////////////////////////////////////////////////
  775. //
  776. // nsInstalledExtensionReader
  777. //
  778. function nsInstalledExtensionReader(aManager)
  779. {
  780.   this._installedExtensions = getFile(KEY_APPDIR,
  781.                                       [DIR_EXTENSIONS, 
  782.                                        FILE_INSTALLED_EXTENSIONS]);
  783.   this._installedExtensionsProcessed = getFile(KEY_APPDIR,
  784.                                                [DIR_EXTENSIONS, 
  785.                                                 FILE_INSTALLED_EXTENSIONS_PROCESSED]);
  786.   this._manager = aManager;
  787. }
  788.  
  789. nsInstalledExtensionReader.prototype = {
  790.   _manager            : null,
  791.   _installedExtensions: null,
  792.   
  793.   read: function nsInstalledExtensionReader_read ()
  794.   {
  795.     if (this._installedExtensionsProcessed.exists())
  796.       return;
  797.     
  798.     if (!this._installedExtensions.exists()) {
  799.       var defaultsList = getFile(KEY_DEFAULTS, [DIR_DEFAULTS_EXTENSIONS, FILE_INSTALLED_EXTENSIONS]);
  800.       defaultsList.copyTo(getDir(KEY_APPDIR, [DIR_EXTENSIONS]), FILE_INSTALLED_EXTENSIONS);
  801.     }
  802.       
  803.     var fis = Components.classes["@mozilla.org/network/file-input-stream;1"]
  804.                         .createInstance(Components.interfaces.nsIFileInputStream);
  805.     fis.init(this._installedExtensions, -1, -1, false);
  806.     var lis = fis.QueryInterface(Components.interfaces.nsILineInputStream);
  807.     var line = { value: "" };
  808.     var more = false;
  809.     var lines = [];
  810.     do {
  811.       more = lis.readLine(line);
  812.       lines.push(line.value);
  813.     }
  814.     while (more);
  815.     fis.close();
  816.  
  817.     // Now that we've closed the stream we can remove all the files    
  818.     for (var i = 0; i < lines.length; ++i)
  819.       this._parseLine(lines[i]);
  820.     
  821.     this._installedExtensions.moveTo(getDir(KEY_APPDIR, [DIR_EXTENSIONS]), 
  822.                                      FILE_INSTALLED_EXTENSIONS_PROCESSED);
  823.   },
  824.   
  825.   TOKEN_EXTENSION : "extension",
  826.   TOKEN_THEME     : "theme",
  827.   
  828.   _parseLine: function nsInstalledExtensionReader__parseLine (aLine)
  829.   {
  830.     // extension,{GUID} or theme,{GUID}
  831.     var parts = aLine.split(",");
  832.     var manifest = getFile(KEY_APPDIR, 
  833.                            [DIR_EXTENSIONS, parts[1], FILE_INSTALL_MANIFEST]);
  834.     if (parts.length != 2)
  835.       return;
  836.       
  837.     if (!manifest.exists()) {
  838.       defaultManifest = defaultFile = getFile(KEY_DEFAULTS, 
  839.                                               [DIR_DEFAULTS_EXTENSIONS, parts[1], FILE_INSTALL_MANIFEST]);
  840.       var extensionDir = getDir(KEY_APPDIR, [DIR_EXTENSIONS, parts[1]]);
  841.       defaultManifest.copyTo(extensionDir, FILE_INSTALL_MANIFEST);
  842.       manifest = getFile(KEY_APPDIR, 
  843.                          [DIR_EXTENSIONS, parts[1], FILE_INSTALL_MANIFEST]);
  844.     }
  845.     switch (parts[0]) {
  846.     case this.TOKEN_EXTENSION:
  847.       this._manager.ensurePreConfiguredItem(parts[1], nsIUpdateItem.TYPE_EXTENSION, manifest);
  848.       break;
  849.     case this.TOKEN_THEME:
  850.       this._manager.ensurePreConfiguredItem(parts[1], nsIUpdateItem.TYPE_THEME, manifest);
  851.       break;
  852.     }
  853.   }
  854. };
  855.  
  856. ///////////////////////////////////////////////////////////////////////////////
  857. //
  858. // nsChromeRegistrationException
  859. //
  860. function nsChromeRegistrationException(aChromePath, aIsProfile, aFunctionName)
  861. {
  862.   this.chromePath = aChromePath;
  863.   this.isProfile = aIsProfile;
  864.   this.functionName = aFunctionName;
  865. }
  866. nsChromeRegistrationException.prototype = {
  867.   chromePath      : null,
  868.   isProfile       : true,
  869.   functionName    : null,
  870.   extensionID     : null
  871. };
  872.  
  873. ///////////////////////////////////////////////////////////////////////////////
  874. //
  875. // nsExtensionInstaller
  876. //
  877. function nsExtensionInstaller (aExtensionDS)
  878. {
  879.   this._extensionDS = aExtensionDS;
  880.  
  881.   this._provTypePackage = gRDF.GetResource(EM_NS("package"));
  882.   this._provTypeSkin    = gRDF.GetResource(EM_NS("skin"));
  883.   this._provTypeLocale  = gRDF.GetResource(EM_NS("locale"));
  884.   this._fileProperty    = gRDF.GetResource(EM_NS("file"));
  885.   this._sourceResource  = gRDF.GetResource("urn:mozilla:install-manifest");
  886. }
  887.  
  888. nsExtensionInstaller.prototype = {
  889.   // Utility services and helpers
  890.   _rdf              : null,
  891.   _writer           : null,
  892.  
  893.   // Extension metadata
  894.   _extensionID      : null,
  895.   _isProfile        : true,
  896.   _extDirKey        : KEY_PROFILEDIR,
  897.   
  898.   // Source and target datasources
  899.   _metadataDS       : null,
  900.   _extensionDS      : null,
  901.   
  902.   // RDF objects and properties
  903.   _provTypePackage  : null,
  904.   _provTypeSkin     : null,
  905.   _provTypeLocale   : null,
  906.   _sourceResource   : null,
  907.   _fileProperty     : null,
  908.   
  909.   install: function nsExtensionInstaller_install (aExtensionID, aIsProfile)
  910.   {
  911.     // Initialize the installer for this extension
  912.     this._extensionID = aExtensionID;
  913.     this._isProfile = aIsProfile;
  914.     this._extDirKey = getDirKey(this._isProfile);
  915.  
  916.     // Create a logger to log install operations for uninstall
  917.     this._writer = new nsInstallLogWriter(this._extensionID, this._isProfile);
  918.     this._writer.open();
  919.     
  920.     // Move files from the staging dir into the extension's final home.
  921.     // This function generates uninstall log files and creates backups of
  922.     // existing files. 
  923.     // XXXben - would like to add exception handling here to test for file
  924.     //          I/O failures on uninstall log so that if there's a crash
  925.     //          and the uninstall log is incorrectly/incompletely written 
  926.     //          we can roll back. It's not critical that we do so right now
  927.     //          since if this throws the extension's chrome is never 
  928.     //          registered. 
  929.     this._installExtensionFiles();
  930.     
  931.     // Load the metadata datasource
  932.     var metadataFile = getFile(this._extDirKey, 
  933.                                [DIR_EXTENSIONS, aExtensionID, FILE_INSTALL_MANIFEST]);
  934.     
  935.     this._metadataDS = getInstallManifest(metadataFile);
  936.     if (!this._metadataDS) return;
  937.     
  938.     // Add metadata for the extension to the global extension metadata set
  939.     this._extensionDS.addItemMetadata(this._extensionID, nsIUpdateItem.TYPE_EXTENSION, 
  940.                                       this._metadataDS, this._isProfile);
  941.     
  942.     // Register chrome packages for files specified in the extension manifest
  943.     try {
  944.       this._registerChromeForExtension();
  945.     }
  946.     catch (e) {
  947.       // Failed to register chrome, for any number of reasons - non-existent 
  948.       // contents.rdf file at the location specified, malformed contents.rdf, 
  949.       // etc. Set the "toBeUninstalled" flag so that the extension is uninstalled
  950.       // properly during the subsequent uninstall pass in 
  951.       // |nsExtensionManager::_finalizeOperations|
  952.  
  953.       this._extensionDS.setItemProperty(this._extensionID, 
  954.                                         this._extensionDS._emR("toBeUninstalled"),
  955.                                         this._extensionDS._emL("true"), this._isProfile,
  956.                                         nsIUpdateItem.TYPE_EXTENSION);
  957.       e.extensionID = this._extensionID;
  958.       showMalformedRegistrationError(e);
  959.     }
  960.     
  961.     this._writer.close();
  962.  
  963.     // Unset the "toBeInstalled" flag
  964.     this._extensionDS.setItemProperty(this._extensionID, 
  965.                                       this._extensionDS._emR("toBeInstalled"),
  966.                                       null, this._isProfile,
  967.                                       nsIUpdateItem.TYPE_EXTENSION);
  968.   },
  969.   
  970.   _installExtensionFiles: function nsExtensionInstaller__installExtensionFiles ()
  971.   {
  972.     var sourceXPI = getFile(this._extDirKey, 
  973.                             [DIR_EXTENSIONS, DIR_TEMP, 
  974.                              this._extensionID, 
  975.                              this._extensionID + ".xpi"]);
  976.     var zipReader = Components.classes["@mozilla.org/libjar/zip-reader;1"]
  977.                               .createInstance(Components.interfaces.nsIZipReader);
  978.     zipReader.init(sourceXPI);
  979.     zipReader.open();
  980.  
  981.     var entries = zipReader.findEntries("*");
  982.     while (entries.hasMoreElements()) {
  983.       var entry = entries.getNext().QueryInterface(Components.interfaces.nsIZipEntry);
  984.       
  985.       var parts = entry.name.split("/");
  986.       var subDirs = [DIR_EXTENSIONS, this._extensionID];
  987.       for (var i = 0; i < parts.length; ++i)
  988.         subDirs.push(parts[i]);
  989.       
  990.       var fileName = parts[parts.length-1];
  991.       if (fileName != "") {
  992.         var targetFile = getFile(this._extDirKey, subDirs);
  993.         zipReader.extract(entry.name, targetFile);
  994.         this._writer.addFile(targetFile.QueryInterface(Components.interfaces.nsILocalFile));
  995.       }
  996.     }
  997.     zipReader.close();
  998.     // Kick off the extraction on a new thread, then join to wait for it to
  999.     // complete. 
  1000.     // (new nsJarFileExtractor(aZipReader.file, dir)).extract();
  1001.     
  1002.     this._cleanUpStagedXPI();
  1003.   },
  1004.   
  1005.   _cleanUpStagedXPI: function nsExtensionInstaller__cleanUpStagedXPI ()
  1006.   {
  1007.     var stageDir = getDir(this._extDirKey, 
  1008.                           [DIR_EXTENSIONS, DIR_TEMP, this._extensionID]);
  1009.     var sourceXPI = stageDir.clone();
  1010.     sourceXPI.append(this._extensionID + ".xpi");
  1011.     sourceXPI.remove(false);
  1012.     
  1013.     // Remove the extension's stage dir
  1014.     if (!stageDir.directoryEntries.hasMoreElements()) 
  1015.       stageDir.remove(false);
  1016.       
  1017.     // If the parent "temp" dir is empty, remove it.
  1018.     try { // XXXben
  1019.       if (!stageDir.parent.directoryEntries.hasMoreElements())
  1020.         stageDir.parent.remove(false);
  1021.     }
  1022.     catch (e) { }
  1023.   },
  1024.   
  1025.   _registerChromeForExtension: function nsExtensionInstaller__registerChromeForExtension ()
  1026.   {
  1027.     // Enumerate the metadata datasource files collection and register chrome
  1028.     // for each file, calling _registerChrome for each.
  1029.     var chromeDir = getDir(this._extDirKey, 
  1030.                            [DIR_EXTENSIONS, this._extensionID, DIR_CHROME]);
  1031.     
  1032.     var files = this._metadataDS.GetTargets(this._sourceResource, this._fileProperty, true);
  1033.     while (files.hasMoreElements()) {
  1034.       var file = files.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  1035.       var chromeFile = chromeDir.clone();
  1036.       var fileName = file.Value.substr("urn:mozilla:extension:file:".length, file.Value.length);
  1037.       chromeFile.append(fileName);
  1038.       
  1039.       var providers = [this._provTypePackage, this._provTypeSkin, this._provTypeLocale];
  1040.       for (var i = 0; i < providers.length; ++i) {
  1041.         var items = this._metadataDS.GetTargets(file, providers[i], true);
  1042.         while (items.hasMoreElements()) {
  1043.           var item = items.getNext().QueryInterface(Components.interfaces.nsIRDFLiteral);
  1044.           this._registerChrome(chromeFile, providers[i], item.Value);
  1045.         }
  1046.       }
  1047.     }
  1048.   },
  1049.   
  1050.   _registerChrome: function nsExtensionInstaller__registerChrome (aFile, aChromeType, aPath)
  1051.   { 
  1052.     var fileURL = getURLSpecFromFile(aFile);
  1053.     if (!aFile.isDirectory()) // .jar files
  1054.       fileURL = "jar:" + fileURL + "!/" + aPath;
  1055.     else                      // flat chrome hierarchies
  1056.       fileURL = fileURL + aPath;
  1057.     
  1058.     var cr = Components.classes["@mozilla.org/chrome/chrome-registry;1"]
  1059.                        .getService(Components.interfaces.nsIXULChromeRegistry);
  1060.     var type;
  1061.     if (aChromeType.EqualsNode(this._provTypePackage)) {
  1062.       try {
  1063.         cr.installPackage(fileURL, this._isProfile);
  1064.       }
  1065.       catch (e) {
  1066.         throw new nsChromeRegistrationException(fileURL, this._isProfile, "installPackage");
  1067.       }
  1068.       type = this._writer.CHROME_TYPE_PACKAGE;
  1069.     }
  1070.     else if (aChromeType.EqualsNode(this._provTypeSkin)) {
  1071.       try {
  1072.         cr.installSkin(fileURL, this._isProfile, true); // Extension skins can execute scripts
  1073.       }
  1074.       catch (e) {
  1075.         throw new nsChromeRegistrationException(fileURL, this._isProfile, "installSkin");
  1076.       }
  1077.       type = this._writer.CHROME_TYPE_SKIN;
  1078.     }
  1079.     else if (aChromeType.EqualsNode(this._provTypeLocale)) {
  1080.       try {
  1081.         cr.installLocale(fileURL, this._isProfile);
  1082.       }
  1083.       catch (e) {
  1084.         throw new nsChromeRegistrationException(fileURL, this._isProfile, "installLocale");
  1085.       }
  1086.       type = this._writer.CHROME_TYPE_LOCALE;
  1087.     }
  1088.     var providerNames = this._getProviderNames(fileURL, type);
  1089.     for (var i = 0; i < providerNames.length; ++i) {
  1090.       this._writer.registerChrome(providerNames[i], type, this._isProfile);
  1091.       
  1092.       // Make sure we enable overlays for this extension so that if it is disabled by
  1093.       // mismatch checking, installing a newer version (as opposed to enabling as a 
  1094.       // result of a version compatibility update) makes the extension's overlaid UI 
  1095.       // appear immediately.
  1096.       cr.setAllowOverlaysForPackage(providerNames[i], true);
  1097.     }
  1098.   },
  1099.   
  1100.   _getProviderNames: function nsExtensionInstaller__getProviderNames (aBaseURL, aType)
  1101.   {
  1102.     if (aBaseURL.charAt(aBaseURL.length-1) != "/")
  1103.       aBaseURL += "/";
  1104.     var manifestURL = aBaseURL + "contents.rdf";
  1105.     
  1106.     var providerNames = [];
  1107.  
  1108.     try {
  1109.       // Discover the list of provider names to register for the location 
  1110.       // specified in the provider arc.
  1111.       //
  1112.       // The contents.rdf file will look like this:
  1113.       //
  1114.       //   <RDF:Seq about="urn:mozilla:<type>:root">
  1115.       //     <RDF:li resource="urn:mozilla:<type>:itemName1"/>
  1116.       //     <RDF:li resource="urn:mozilla:<type>:itemName2"/>
  1117.       //     ..
  1118.       //   </RDF:Seq>
  1119.       //
  1120.       // We need to explicitly walk this list here, we don't need to do so
  1121.       // for nsIXULChromeRegistry's |installPackage| method since that does
  1122.       // this same thing itself.
  1123.       
  1124.       var ds = gRDF.GetDataSourceBlocking(manifestURL);
  1125.       
  1126.       var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  1127.                           .createInstance(Components.interfaces.nsIRDFContainer);
  1128.       ctr.Init(ds, gRDF.GetResource("urn:mozilla:" + aType + ":root"));
  1129.       
  1130.       var items = ctr.GetElements();
  1131.       while (items.hasMoreElements()) {
  1132.         var item = items.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  1133.         
  1134.         var nameArc = gRDF.GetResource(CHROME_NS("name"));
  1135.         var name;
  1136.         if (ds.hasArcOut(item, nameArc))
  1137.           name = stringData(ds.GetTarget(item, nameArc, true));
  1138.         else {
  1139.           var parts = item.Value.split(":");
  1140.           name = parts[parts.length-1];
  1141.         }
  1142.         providerNames.push(name);
  1143.       }
  1144.     }
  1145.     catch (e) { }
  1146.     
  1147.     return providerNames;
  1148.   }
  1149. };
  1150.  
  1151. ///////////////////////////////////////////////////////////////////////////////
  1152. //
  1153. // nsExtensionUninstaller
  1154. //
  1155. function nsExtensionUninstaller(aExtensionDS)
  1156. {
  1157.   this._cr = Components.classes["@mozilla.org/chrome/chrome-registry;1"]
  1158.                        .getService(Components.interfaces.nsIXULChromeRegistry);
  1159.   this._extensionDS = aExtensionDS;
  1160. }
  1161.  
  1162. nsExtensionUninstaller.prototype = {
  1163.   _extensionDS  : null,
  1164.   _cr           : null,
  1165.   _isProfile    : true,
  1166.   _extDirKey    : "",
  1167.   _extensionsDir: null,
  1168.   _extensionID  : "",
  1169.  
  1170.   uninstall: function nsExtensionUninstaller_uninstall (aExtensionID, aIsProfile)
  1171.   {
  1172.     // Initialize the installer for this extension
  1173.     this._extensionID = aExtensionID;
  1174.     this._isProfile = aIsProfile;
  1175.     this._extDirKey = getDirKey(this._isProfile);
  1176.     this._extensionsDir = getDir(this._extDirKey, [DIR_EXTENSIONS]);
  1177.  
  1178.     // Create a logger to log install operations for uninstall
  1179.     this._reader = new nsInstallLogReader(this._extensionID, 
  1180.                                           this._isProfile, 
  1181.                                           this);
  1182.     try { // XXXben don't let errors stop us. 
  1183.       this._reader.read();
  1184.       
  1185.       // Now remove the uninstall log file. 
  1186.       this._removeFile(this._reader.uninstallLog);
  1187.     }
  1188.     catch (e) {
  1189.       dump("******* Failed to remove extension uninstall log, with exception = " + e + "\n");
  1190.     }
  1191.     
  1192.     // Unset the "toBeUninstalled" flag
  1193.     this._extensionDS.setItemProperty(this._extensionID, 
  1194.                                       this._extensionDS._emR("toBeUninstalled"),
  1195.                                       null, this._isProfile,
  1196.                                       nsIUpdateItem.TYPE_EXTENSION);
  1197.   },
  1198.   
  1199.   ///////////////////////////////////////////////////////////////////////////////
  1200.   // nsIInstallLogReaderListener
  1201.   onAddFile: function nsExtensionUninstaller_onAddFile (aFile)
  1202.   {
  1203.     this._removeFile(aFile);
  1204.   },
  1205.   
  1206.   _removeFile: function nsExtensionUninstaller__removeFile (aFile)
  1207.   {
  1208.     if (aFile.exists()) {
  1209.       aFile.remove(false);
  1210.       
  1211.       // Clean up the parent hierarchy if possible  
  1212.       var parent = aFile.parent;
  1213.       var e = parent.directoryEntries;
  1214.       if (!e.hasMoreElements() && 
  1215.           !parent.equals(this._extensionsDir)) // stop at the extensions dir
  1216.         this._removeFile(parent);
  1217.     }
  1218.   },
  1219.   
  1220.   // XXXben - maybe we should find a way to 
  1221.   _packagesForExtension: [],
  1222.   
  1223.   onRegisterChrome: function nsExtensionUninstaller_onRegisterChrome (aProviderName, aFile, aChromeType, aIsProfile)
  1224.   {
  1225.     switch (aChromeType) {
  1226.     case this._reader.CHROME_TYPE_PACKAGE:
  1227.       this._packagesForExtension.push(aProviderName);
  1228.       this._cr.uninstallPackage(aProviderName, aIsProfile)
  1229.       break;
  1230.     case this._reader.CHROME_TYPE_SKIN:
  1231.       for (var i = 0; i < this._packagesForExtension.length; ++i) {
  1232.         this._cr.deselectSkinForPackage(aProviderName, 
  1233.                                         this._packagesForExtension[i], 
  1234.                                         aIsProfile);
  1235.       }
  1236.       // this._cr.uninstallSkin(aProviderName, aIsProfile)
  1237.       break;
  1238.     case this._reader.CHROME_TYPE_LOCALE:
  1239.       for (var i = 0; i < this._packagesForExtension.length; ++i) {
  1240.         this._cr.deselectLocaleForPackage(aProviderName, 
  1241.                                           this._packagesForExtension[i], 
  1242.                                           aIsProfile);
  1243.       }
  1244.       // this._cr.uninstallLocale(aProviderName, aIsProfile)
  1245.       break;
  1246.     }
  1247.   }
  1248. };
  1249.  
  1250. ///////////////////////////////////////////////////////////////////////////////
  1251. //
  1252. // nsExtensionEnabler
  1253. //
  1254. function nsExtensionEnabler(aExtensionDS)
  1255. {
  1256.   this._cr = Components.classes["@mozilla.org/chrome/chrome-registry;1"]
  1257.                        .getService(Components.interfaces.nsIXULChromeRegistry);
  1258.   this._extensionDS = aExtensionDS;
  1259. }
  1260.  
  1261. nsExtensionEnabler.prototype = {
  1262.   _extensionDS  : null,
  1263.   _cr           : null,
  1264.   _enable       : true,
  1265.   _isProfile    : true,
  1266.   _extDirKey    : "",
  1267.   _extensionsDir: null,
  1268.  
  1269.   enable: function nsExtensionEnabler_enable (aExtensionID, aIsProfile, aDisable)
  1270.   {
  1271.     // Initialize the installer for this extension
  1272.     this._enable = !aDisable;
  1273.     this._extensionID = aExtensionID;
  1274.     this._isProfile = aIsProfile;
  1275.     this._extDirKey = getDirKey(this._isProfile);
  1276.     this._extensionsDir = getDir(this._extDirKey, [DIR_EXTENSIONS]);
  1277.  
  1278.     // Create a logger to log install operations for uninstall
  1279.     this._reader = new nsInstallLogReader(this._extensionID, 
  1280.                                           this._isProfile, 
  1281.                                           this);
  1282.     this._reader.read();
  1283.   },
  1284.   
  1285.   onRegisterChrome: function nsExtensionEnabler_onRegisterChrome (aProviderName, aFile, aChromeType, aIsProfile)
  1286.   {
  1287.     if (aChromeType == this._reader.CHROME_TYPE_PACKAGE)
  1288.       this._cr.setAllowOverlaysForPackage(aProviderName, this._enable);
  1289.   },
  1290.   
  1291.   onAddFile: function nsExtensionEnabler_onAddFile (aFile)
  1292.   {
  1293.   }
  1294. };
  1295.  
  1296. ///////////////////////////////////////////////////////////////////////////////
  1297. //
  1298. // nsThemeInstaller
  1299. //
  1300. function nsThemeInstaller(aExtensionDS, aManager)
  1301. {
  1302.   this._extensionDS = aExtensionDS;
  1303.   this._em = aManager;
  1304. }
  1305.  
  1306. nsThemeInstaller.prototype = {
  1307.   _extensionDS  : null,
  1308.   _isProfile    : true,
  1309.   _extDirKey    : "",
  1310.  
  1311.   install: function nsThemeInstaller_install (aJARFile, aIsProfile)
  1312.   {
  1313.     var extDirKey = getDirKey(aIsProfile);
  1314.  
  1315.     // Since we're installing a "new type" theme, we assume a file layout
  1316.     // within the JAR like so:
  1317.     // foo.jar/
  1318.     //         install.rdf      <-- Theme Manager metadata
  1319.     //         contents.rdf   <-- Chrome Registry metadata
  1320.     //         browser/
  1321.     //         global/
  1322.     //         ...
  1323.     var zipReader = Components.classes["@mozilla.org/libjar/zip-reader;1"]
  1324.                               .createInstance(Components.interfaces.nsIZipReader);
  1325.     zipReader.init(aJARFile);
  1326.     zipReader.open();
  1327.     
  1328.     try {
  1329.       zipReader.getEntry(FILE_INSTALL_MANIFEST);
  1330.     }
  1331.     catch (e) {
  1332.       // If the zip reader returned an error code here it means that the install.rdf
  1333.       // file was not found in the theme jar file - i.e. it was an old style theme. 
  1334.       // There's no reason for people to be installing or maintaining such themes 
  1335.       // anymore as there is no practical use for them, so we should throw an error
  1336.       // here and bail. 
  1337.       try {
  1338.         zipReader.getEntry(FILE_CHROME_MANIFEST);
  1339.         
  1340.         // Load the contents.rdf file from the .jar file if present and show a detailed
  1341.         // error.         
  1342.         var tempChromeManifest = getFile(extDirKey,
  1343.                                          [DIR_EXTENSIONS, DIR_TEMP, FILE_CHROME_MANIFEST]);
  1344.         zipReader.extract(FILE_CHROME_MANIFEST, tempChromeManifest);
  1345.         var rdfs = Components.classes["@mozilla.org/rdf/rdf-service;1"]
  1346.                              .getService(Components.interfaces.nsIRDFService);
  1347.         showOldThemeError(rdfs.GetDataSourceBlocking(getURLSpecFromFile(tempChromeManifest)));
  1348.         tempChromeManifest.remove(false);
  1349.       }
  1350.       catch (e) {
  1351.         showMissingFileError(aJARFile, FILE_CHROME_MANIFEST);
  1352.       }
  1353.     }
  1354.     
  1355.     var themeManifest = getFile(extDirKey,
  1356.                                 [DIR_EXTENSIONS, DIR_TEMP, getRandomFileName("install", "rdf")]);
  1357.     zipReader.extract(FILE_INSTALL_MANIFEST, themeManifest);
  1358.     
  1359.     var chromeManifest = getFile(extDirKey,
  1360.                                  [DIR_EXTENSIONS, DIR_TEMP, FILE_CHROME_MANIFEST]);
  1361.     zipReader.extract(FILE_CHROME_MANIFEST, chromeManifest);
  1362.     
  1363.     var themeMetadata = getInstallManifest(themeManifest);
  1364.     if (!themeMetadata) return;
  1365.     var chromeMetadata = gRDF.GetDataSourceBlocking(getURLSpecFromFile(chromeManifest));
  1366.     
  1367.     // We do a basic version check first just to make sure we somehow weren't 
  1368.     // tricked into installing an incompatible theme...
  1369.     this._themeID = this._em.canInstallItem(themeMetadata);
  1370.     if (isNaN(parseInt(this._themeID))) {
  1371.       var canInstall = true;
  1372.  
  1373.       // Copy the file to its final location
  1374.       var destinationDir = getDir(extDirKey, 
  1375.                                   [DIR_EXTENSIONS, this._themeID, DIR_CHROME]);
  1376.       var destinationFile = destinationDir.clone();
  1377.       destinationFile.append(aJARFile.leafName);
  1378.       if (destinationFile.exists())
  1379.         destinationFile.remove(false);
  1380.       aJARFile.copyTo(destinationDir, aJARFile.leafName);
  1381.  
  1382.       var nameArc = gRDF.GetResource(CHROME_NS("name"));
  1383.       var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  1384.                           .createInstance(Components.interfaces.nsIRDFContainer);
  1385.       ctr.Init(chromeMetadata, gRDF.GetResource("urn:mozilla:skin:root"));
  1386.       
  1387.       var elts = ctr.GetElements();
  1388.       while (elts.hasMoreElements()) {
  1389.         var elt = elts.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  1390.         var chromeSkinPrefix = "urn:mozilla:skin:";
  1391.         if (elt.Value.substr(0, chromeSkinPrefix.length) == chromeSkinPrefix) {
  1392.           var name = chromeMetadata.GetTarget(elt, nameArc, true);
  1393.           
  1394.           // Check to see if the em:internalName property on the theme install
  1395.           // manifest matches the chrome:name property on the theme's CR entry. 
  1396.           var manifestRoot = gRDF.GetResource("urn:mozilla:install-manifest");
  1397.           var internalName = themeMetadata.GetTarget(manifestRoot, 
  1398.                                                      gRDF.GetResource(EM_NS("internalName")),
  1399.                                                      true);
  1400.           if (!internalName.EqualsNode(name)) {
  1401.             var consoleService = Components.classes["@mozilla.org/consoleservice;1"]
  1402.                                            .getService(Components.interfaces.nsIConsoleService);
  1403.             consoleService.logStringMessage("Could not install theme because chrome:name arc in " +
  1404.                                             "the theme's contents.rdf file (" + stringData(name) +
  1405.                                             ") does not match the em:internalName arc in the theme's " + 
  1406.                                             "install.rdf file (" + stringData(internalName) + ")");
  1407.             var fileURL = getURLSpecFromFile(themeManifest);
  1408.             var uri = Components.classes["@mozilla.org/network/standard-url;1"]
  1409.                                 .createInstance(Components.interfaces.nsIURI);
  1410.             uri.spec = fileURL;
  1411.             var url = uri.QueryInterface(Components.interfaces.nsIURL);
  1412.             showMalformedError(url.fileName);
  1413.             
  1414.             destinationFile.remove(false);
  1415.             destinationDir.remove(true);
  1416.             this._em._cleanDirs();
  1417.             
  1418.             canInstall = false;
  1419.           }
  1420.  
  1421.           if (canInstall) {
  1422.             name = name.QueryInterface(Components.interfaces.nsIRDFLiteral).Value;
  1423.  
  1424.             // Create a logger to log install operations for uninstall
  1425.             this._writer = new nsInstallLogWriter(this._themeID, aIsProfile);
  1426.             this._writer.open();
  1427.             this._writer.installSkin(name, aIsProfile);
  1428.           }
  1429.         }
  1430.       }
  1431.  
  1432.       if (canInstall) {
  1433.         this._writer.addFile(destinationFile.QueryInterface(Components.interfaces.nsILocalFile));
  1434.         this._writer.close();
  1435.  
  1436.         // Use the Chrome Registry API to install the theme there
  1437.         var filePath = "jar:" + getURLSpecFromFile(destinationFile) + "!/";      
  1438.         var cr = Components.classes["@mozilla.org/chrome/chrome-registry;1"]
  1439.                           .getService(Components.interfaces.nsIXULChromeRegistry);
  1440.         cr.installSkin(filePath, aIsProfile, false);
  1441.  
  1442.         // Insert the theme into the theme list. 
  1443.         this._extensionDS.insertForthcomingItem(this._themeID, nsIUpdateItem.TYPE_THEME, 
  1444.                                                 aIsProfile);
  1445.  
  1446.         // Add metadata for the extension to the global extension metadata set
  1447.         this._extensionDS.addItemMetadata(this._themeID, nsIUpdateItem.TYPE_THEME,
  1448.                                           themeMetadata, aIsProfile);
  1449.       }
  1450.       
  1451.       this._extensionDS.doneInstallingTheme(this._themeID);
  1452.     }
  1453.     else if (this._themeID == 0)
  1454.       showIncompatibleError(themeMetadata);
  1455.     
  1456.     zipReader.close();
  1457.     themeManifest.remove(false);
  1458.     chromeManifest.remove(false);
  1459.   }
  1460. };
  1461.  
  1462. ///////////////////////////////////////////////////////////////////////////////
  1463. //
  1464. // nsThemeUninstaller
  1465. //
  1466. function nsThemeUninstaller(aExtensionDS)
  1467. {
  1468.   this._cr = Components.classes["@mozilla.org/chrome/chrome-registry;1"]
  1469.                        .getService(Components.interfaces.nsIXULChromeRegistry);
  1470. }
  1471.  
  1472. nsThemeUninstaller.prototype = {
  1473.   _extensionsDir : null,
  1474.   
  1475.   uninstall: function nsThemeUninstaller_uninstall (aThemeID, aIsProfile)
  1476.   {
  1477.     this._extensionsDir = getDir(getDirKey(aIsProfile), [DIR_EXTENSIONS]);
  1478.  
  1479.     // Create a logger to log install operations for uninstall
  1480.     this._reader = new nsInstallLogReader(aThemeID, aIsProfile, this);
  1481.     try { // XXXben don't let errors stop us. 
  1482.       this._reader.read();
  1483.       
  1484.       // Now remove the uninstall log file. 
  1485.       this._removeFile(this._reader.uninstallLog);
  1486.     }
  1487.     catch (e) {
  1488.       dump("******* Failed to remove theme uninstall log, with exception = " + e + "\n");
  1489.     }
  1490.   },
  1491.   
  1492.   ///////////////////////////////////////////////////////////////////////////////
  1493.   // nsIInstallLogReaderListener
  1494.   onAddFile: function nsThemeUninstaller_onAddFile (aFile)
  1495.   {
  1496.     this._removeFile(aFile);
  1497.   },
  1498.   
  1499.   _removeFile: function nsThemeUninstaller__removeFile (aFile)
  1500.   {
  1501.     if (aFile.exists()) {
  1502.       aFile.remove(false);
  1503.       
  1504.       // Clean up the parent hierarchy if possible  
  1505.       var parent = aFile.parent;
  1506.       var e = parent.directoryEntries;
  1507.       if (!e.hasMoreElements() && 
  1508.           !parent.equals(this._extensionsDir)) // stop at the extensions dir
  1509.         this._removeFile(parent);
  1510.     }
  1511.   },
  1512.   
  1513.   onInstallSkin: function nsThemeUninstaller_onInstallSkin (aSkinName, aIsProfile)
  1514.   {
  1515.     this._cr.uninstallSkin(aSkinName, aIsProfile);
  1516.   }
  1517. };
  1518.  
  1519. ///////////////////////////////////////////////////////////////////////////////
  1520. //
  1521. // nsExtensionManager
  1522. //
  1523. function nsExtensionManager()
  1524. {
  1525.   gPref = Components.classes["@mozilla.org/preferences-service;1"]
  1526.                     .getService(Components.interfaces.nsIPrefBranch);
  1527.   gRDF = Components.classes["@mozilla.org/rdf/rdf-service;1"]
  1528.                    .getService(Components.interfaces.nsIRDFService);
  1529.   gOS = Components.classes["@mozilla.org/observer-service;1"]
  1530.                   .getService(Components.interfaces.nsIObserverService);
  1531.  
  1532.   gOS.addObserver(this, "xpcom-shutdown", false);
  1533.  
  1534.   ensureExtensionsFiles(false);
  1535. }
  1536.  
  1537. nsExtensionManager.prototype = {
  1538.   _extInstaller     : null,
  1539.   _extUninstaller   : null,
  1540.   _extEnabler       : null,
  1541.   _started          : false,
  1542.   
  1543.   /////////////////////////////////////////////////////////////////////////////
  1544.   // nsIObserver
  1545.   observe: function nsExtensionManager_observe (aSubject, aTopic, aData)
  1546.   {
  1547.     switch (aTopic) {
  1548.     case "quit-application-requested":
  1549.       if (this._downloadCount > 0) {
  1550.         var result;
  1551.         result = this._confirmCancelDownloads(this._downloadCount, 
  1552.                                               "quitCancelDownloadsAlertTitle",
  1553.                                               "quitCancelDownloadsAlertMsgMultiple",
  1554.                                               "quitCancelDownloadsAlertMsg",
  1555.                                               "dontQuitButtonWin");
  1556.         if (!result)
  1557.           this._cancelDownloads();
  1558.         var PRBool = aSubject.QueryInterface(Components.interfaces.nsISupportsPRBool);
  1559.         PRBool.data = result;
  1560.       }
  1561.       break;
  1562.     case "offline-requested":
  1563.       if (this._downloadCount > 0) {
  1564.         result = this._confirmCancelDownloads(this._downloadCount,
  1565.                                               "offlineCancelDownloadsAlertTitle",
  1566.                                               "offlineCancelDownloadsAlertMsgMultiple",
  1567.                                               "offlineCancelDownloadsAlertMsg",
  1568.                                               "dontGoOfflineButton");
  1569.         if (!result)
  1570.           this._cancelDownloads();
  1571.         var PRBool = aSubject.QueryInterface(Components.interfaces.nsISupportsPRBool);
  1572.         PRBool.data = result;
  1573.       }
  1574.       break;  
  1575.     case "xpcom-shutdown":
  1576.       gOS.removeObserver(this, "xpcom-shutdown");    
  1577.  
  1578.       // Release strongly held services.
  1579.       gPref           = null;
  1580.       gRDF            = null;
  1581.       gOS             = null;
  1582.       gVersionChecker = null;
  1583.       break;
  1584.     }
  1585.   },
  1586.   
  1587.   start: function nsExtensionManager_start (aIsDirty)
  1588.   {
  1589.     this._started = true;
  1590.     
  1591.     var needsRestart = false;
  1592.   
  1593.     ensureExtensionsFiles(true);
  1594.     
  1595.     // Somehow the component list went away, and for that reason the new one
  1596.     // generated by this function is going to result in a different compreg.
  1597.     // We must force a restart.
  1598.     var componentList = getFile(KEY_PROFILEDIR, [FILE_COMPONENT_MANIFEST]);
  1599.     if (!componentList.exists())
  1600.       needsRestart = true;
  1601.     
  1602.     // XXXben - a bit of a hack - clean up any empty dirs that may not have been
  1603.     //          properly removed by [un]install... I should really investigate those
  1604.     //          cases to see what is stopping these dirs from being removed, but no
  1605.     //          time now.
  1606.     this._cleanDirs();
  1607.   
  1608.     var cmdLineSvc = Components.classes["@mozilla.org/appshell/commandLineService;1"]
  1609.                                 .getService(Components.interfaces.nsICmdLineService);
  1610.     var safeMode = cmdLineSvc.getCmdLineValue("-safe-mode") != null;
  1611.     if (!safeMode) {
  1612.       var wasInSafeModeFile = getFile(KEY_PROFILEDIR, [DIR_EXTENSIONS, FILE_WASINSAFEMODE]);
  1613.       if (wasInSafeModeFile.exists()) {
  1614.         // Clean up after we were in safe mode
  1615.         var win = this._showProgressWindow();
  1616.         try {
  1617.           this._ensureDS();
  1618.           
  1619.           // Retrieve the skin that was selected prior to entering safe mode
  1620.           // and select it. 
  1621.           var lastSelectedSkin = KEY_DEFAULT_THEME;
  1622.           try {
  1623.             lastSelectedSkin = gPref.getCharPref(PREF_EM_LAST_SELECTED_SKIN);
  1624.             gPref.clearUserPref(PREF_EM_LAST_SELECTED_SKIN);
  1625.             gPref.setCharPref(PREF_GENERAL_SKINS_SELECTEDSKIN, lastSelectedSkin);
  1626.           } 
  1627.           catch (e) { }
  1628.           
  1629.           // Walk the list of extensions and re-activate overlays for packages 
  1630.           // that aren't disabled.
  1631.           var items = this._ds.getItemsWithFlagUnset("disabled", nsIUpdateItem.TYPE_EXTENSION);
  1632.           for (var i = 0; i < items.length; ++i)
  1633.             this._finalizeEnableDisable(items[i], false);
  1634.             
  1635.           wasInSafeModeFile.remove(false);
  1636.           
  1637.           this._writeDefaults(true);
  1638.           try {
  1639.             this._writeDefaults(false);
  1640.           }
  1641.           catch (e) { }
  1642.         }
  1643.         catch (e) {
  1644.           dump("*** nsExtensionManager::start - failure, catching exception so finalize window can close = " + e + "\n");
  1645.         }
  1646.         win.close();
  1647.         
  1648.         needsRestart = true;
  1649.       }
  1650.  
  1651.       if (aIsDirty) 
  1652.         needsRestart = this._finishOperations();
  1653.     }
  1654.     else {
  1655.       var win = this._showProgressWindow();
  1656.       try {    
  1657.         // Enter safe mode
  1658.         this._ensureDS();
  1659.  
  1660.         // Save the current theme (assumed to be the theme that styles the global
  1661.         // package) and re-select the default theme ("classic/1.0")
  1662.         if (!gPref.prefHasUserValue(PREF_EM_LAST_SELECTED_SKIN)) {
  1663.           gPref.setCharPref(PREF_EM_LAST_SELECTED_SKIN,
  1664.                             gPref.getCharPref(PREF_GENERAL_SKINS_SELECTEDSKIN));
  1665.           if (gPref.prefHasUserValue(PREF_GENERAL_SKINS_SELECTEDSKIN))
  1666.             gPref.clearUserPref(PREF_GENERAL_SKINS_SELECTEDSKIN);
  1667.         }
  1668.  
  1669.         var items = this._ds.getItemList(null, nsIUpdateItem.TYPE_EXTENSION, {});
  1670.         for (var i = 0; i < items.length; ++i)
  1671.           this._finalizeEnableDisable(items[i].id, true);
  1672.           
  1673.         this._ds.safeMode = true;
  1674.         
  1675.         this._writeDefaults(true);
  1676.         try {
  1677.           this._writeDefaults(false);
  1678.         }
  1679.         catch (e) { }
  1680.  
  1681.         needsRestart = true;
  1682.  
  1683.         var wasInSafeModeFile = getFile(KEY_PROFILEDIR, [DIR_EXTENSIONS, FILE_WASINSAFEMODE]);
  1684.         if (!wasInSafeModeFile.exists())
  1685.           wasInSafeModeFile.create(Components.interfaces.nsIFile.NORMAL_FILE_TYPE, 0644);
  1686.         else {
  1687.           // If the "Safe Mode" file already exists, then we are in the second launch of an
  1688.           // app launched with -safe-mode and so we don't want to provoke any further
  1689.           // restarts or re-create the file, just continue starting normally.
  1690.           needsRestart = false;
  1691.         }
  1692.       }
  1693.       catch (e) {
  1694.         dump("*** nsExtensionManager::start - (safe mode) failure, catching exception so finalize window can close = " + e + "\n");
  1695.       }
  1696.       win.close();
  1697.       
  1698.     }
  1699.     return needsRestart;
  1700.   },
  1701.   
  1702.   handleCommandLineArgs: function nsExtensionManager_handleCommandLineArgs ()
  1703.   {
  1704.     var cmdLineSvc = Components.classes["@mozilla.org/appshell/commandLineService;1"]
  1705.                               .getService(Components.interfaces.nsICmdLineService);
  1706.     var globalExtension = cmdLineSvc.getCmdLineValue("-install-global-extension");
  1707.     if (globalExtension)
  1708.       this._checkForGlobalInstalls(globalExtension, nsIUpdateItem.TYPE_EXTENSION);
  1709.       
  1710.     var globalTheme = cmdLineSvc.getCmdLineValue("-install-global-theme");
  1711.     if (globalTheme)
  1712.       this._checkForGlobalInstalls(globalTheme, nsIUpdateItem.TYPE_THEME);
  1713.     
  1714.     var showList = cmdLineSvc.getCmdLineValue("-list-global-items");
  1715.     if (showList)
  1716.       this._showGlobalItemList();
  1717.       
  1718.     var locked = cmdLineSvc.getCmdLineValue("-lock-item");
  1719.     if (locked) {
  1720.       this._ensureDS();
  1721.       this._ds.lockUnlockItem(locked, true);
  1722.     }
  1723.  
  1724.     var unlocked = cmdLineSvc.getCmdLineValue("-unlock-item");
  1725.     if (unlocked) {
  1726.       this._ensureDS();
  1727.       this._ds.lockUnlockItem(unlocked, false);
  1728.     }
  1729.     
  1730.     this._finishOperations();
  1731.   },
  1732.   
  1733.   register: function nsExtensionManager_register ()
  1734.   {
  1735.     if (!this._started) {
  1736.       // Loads the datasource and installs any pre-configured items.
  1737.       this._ds = new nsExtensionsDataSource();
  1738.       this._ds.loadExtensions(false);
  1739.       
  1740.       // Write bin/extensions/Extensions.rdf
  1741.       //       bin/extensions/installed-extensions-processed.txt
  1742.       (new nsInstalledExtensionReader(this)).read();
  1743.  
  1744.       // Write bin/components.ini
  1745.       var manifest = getFile(KEY_APPDIR, [FILE_COMPONENT_MANIFEST]);
  1746.       this._writeProfileFile(manifest, this._getComponentsDir, false);
  1747.     }
  1748.   },
  1749.  
  1750.   _cancelDownloads: function nsExtensionManager__cancelDownloads ()
  1751.   {
  1752.     for (var i = 0; i < this._transactions.length; ++i)
  1753.       gOS.notifyObservers(this._transactions[i], "xpinstall-progress", "cancel");
  1754.     gOS.removeObserver(this, "offline-requested");
  1755.     gOS.removeObserver(this, "quit-application-requested");
  1756.  
  1757.     this._removeAllDownloads();
  1758.   },
  1759.  
  1760.   _confirmCancelDownloads: function nsExtensionManager__confirmCancelDownloads(aCount, 
  1761.     aTitle, aCancelMessageMultiple, aCancelMessageSingle, aDontCancelButton)
  1762.   {
  1763.     var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  1764.                         .getService(Components.interfaces.nsIStringBundleService);
  1765.     var bundle = sbs.createBundle("chrome://mozapps/locale/downloads/downloads.properties");
  1766.     var title = bundle.GetStringFromName(aTitle);
  1767.     var message, quitButton;
  1768.     if (aCount > 1) {
  1769.       message = bundle.formatStringFromName(aCancelMessageMultiple, [aCount], 1);
  1770.       quitButton = bundle.formatStringFromName("cancelDownloadsOKTextMultiple", [aCount], 1);
  1771.     }
  1772.     else {
  1773.       message = bundle.GetStringFromName(aCancelMessageSingle);
  1774.       quitButton = bundle.GetStringFromName("cancelDownloadsOKText");
  1775.     }
  1776.     var dontQuitButton = bundle.GetStringFromName(aDontCancelButton);
  1777.     
  1778.     var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
  1779.                        .getService(Components.interfaces.nsIWindowMediator);
  1780.     var win = wm.getMostRecentWindow("Extension:Manager");
  1781.     const nsIPromptService = Components.interfaces.nsIPromptService;
  1782.     var ps = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
  1783.                        .getService(nsIPromptService);
  1784.     var flags = (nsIPromptService.BUTTON_TITLE_IS_STRING * nsIPromptService.BUTTON_POS_0) +
  1785.                 (nsIPromptService.BUTTON_TITLE_IS_STRING * nsIPromptService.BUTTON_POS_1);
  1786.     var rv = ps.confirmEx(win, title, message, flags, quitButton, dontQuitButton, null, null, { });
  1787.     return rv == 1;
  1788.   },
  1789.   
  1790.   // This function checks for and disables any "old-style" extensions 
  1791.   // from Firefox 0.8 and earlier created using the "chrome:extension=true" flag. 
  1792.   _disableObsoleteExtensions: function nsExtensionManager__disableObsoleteExtensions ()
  1793.   {
  1794.     if (!gPref.prefHasUserValue(PREF_EM_DISABLEDOBSOLETE) || !gPref.getBoolPref(PREF_EM_DISABLEDOBSOLETE)) {
  1795.       var win = this._showProgressWindow();
  1796.       try {
  1797.         var cr = Components.classes["@mozilla.org/chrome/chrome-registry;1"]
  1798.                           .getService(Components.interfaces.nsIXULChromeRegistry);
  1799.         var crDS = gRDF.GetDataSource("rdf:chrome");
  1800.         var disabled = false;
  1801.         var sources = crDS.GetSources(gRDF.GetResource(CHROME_NS("extension")), gRDF.GetLiteral("true"), true);
  1802.         while (sources.hasMoreElements()) {
  1803.           disabled = true;
  1804.           
  1805.           var source = sources.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  1806.           var name = crDS.GetTarget(source, gRDF.GetResource(CHROME_NS("name")), true);
  1807.           if (name) {
  1808.             name = name.QueryInterface(Components.interfaces.nsIRDFLiteral).Value;
  1809.             cr.setAllowOverlaysForPackage(name, false);
  1810.           }
  1811.         }
  1812.  
  1813.         // Re-select the default theme to prevent any incompatibilities with old-style
  1814.         // themes.
  1815.         cr.selectSkin(KEY_DEFAULT_THEME, true);
  1816.       }
  1817.       catch (e) {
  1818.         dump("*** nsExtensionManager::_disableObsoleteExtensions - failure, catching exception so finalize window can close\n");
  1819.       }
  1820.       win.close();
  1821.       
  1822.       if (disabled) {
  1823.         const nsIPromptService = Components.interfaces.nsIPromptService;
  1824.         var ps = Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
  1825.                            .getService(nsIPromptService);
  1826.         var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  1827.                             .getService(Components.interfaces.nsIStringBundleService);
  1828.         var bundle = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  1829.         var title = bundle.GetStringFromName("disabledObsoleteTitle");
  1830.         var message = bundle.GetStringFromName("disabledObsoleteMessage");
  1831.         ps.alert(null, title, message);
  1832.       }
  1833.             
  1834.       gPref.setBoolPref(PREF_EM_DISABLEDOBSOLETE, true);
  1835.     }
  1836.   },
  1837.   
  1838.   _checkForGlobalInstalls: function nsExtensionManager__checkForGlobalInstalls (aPath, aItemType)
  1839.   {
  1840.  
  1841.     // First see if the path supplied is a file path
  1842.     var file = Components.classes["@mozilla.org/file/local;1"]
  1843.                          .createInstance(Components.interfaces.nsILocalFile);
  1844.     try {
  1845.       file.initWithPath(aPath);
  1846.     }
  1847.     catch (e) {
  1848.       // Try appending the path to the current proc dir. 
  1849.       file = getDir(KEY_APPDIR, []);
  1850.       try {
  1851.         file.append(aPath);
  1852.       }
  1853.       catch (e) { /* can't handle this */ }
  1854.     }
  1855.     
  1856.     if (file.exists()) {
  1857.       if (aItemType & nsIUpdateItem.TYPE_EXTENSION)
  1858.         this.installExtension(file, nsIExtensionManager.FLAG_INSTALL_GLOBAL);
  1859.       else if (aItemType & nsIUpdateItem.TYPE_THEME)
  1860.         this.installTheme(file, nsIExtensionManager.FLAG_INSTALL_GLOBAL);
  1861.     }
  1862.     else
  1863.       dump("Invalid XPI/JAR Path: " + aPath + "\n");
  1864.   },
  1865.   
  1866.   _showGlobalItemList: function nsExtensionManager__showGlobalItemList ()
  1867.   {
  1868.     this._ensureDS();
  1869.     
  1870.     var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  1871.                         .getService(Components.interfaces.nsIStringBundleService);
  1872.     var bundle = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  1873.  
  1874.     dump(bundle.GetStringFromName("globalItemList"));
  1875.     dump(bundle.GetStringFromName("globalItemListExtensions"));
  1876.     var items = this.getItemList(null, nsIUpdateItem.TYPE_EXTENSION, {});
  1877.     for (var i = 0; i < items.length; ++i)
  1878.       dump(" " + items[i].id + "   " + items[i].name + " " + items[i].version + "\n");
  1879.     dump(bundle.GetStringFromName("globalItemListThemes"));
  1880.     items = this.getItemList(null, nsIUpdateItem.TYPE_THEME, {});
  1881.     for (var i = 0; i < items.length; ++i)
  1882.       dump(" " + items[i].id + "   " + items[i].name + " " + items[i].version + "\n");
  1883.       
  1884.     dump("\n\n");
  1885.   },
  1886.   
  1887.   _finishOperations: function nsExtensionManager__finishOperations ()
  1888.   {
  1889.     var win = this._showProgressWindow();
  1890.   
  1891.     try {
  1892.       // An existing autoreg file is an indication that something major has 
  1893.       // happened to the extensions datasource (install/uninstall/enable/disable)
  1894.       // and as such we must load it now and see what needs to happen.
  1895.       this._ensureDS();
  1896.       
  1897.       // Look for items that need to be installed
  1898.       var items = this._ds.getItemsWithFlagSet("toBeInstalled");
  1899.       for (var i = 0; i < items.length; ++i)
  1900.         this._finalizeInstall(items[i]);
  1901.  
  1902.       // If there were any install operations, we need to restart (again!) after 
  1903.       // the component files have been properly installed are registered...
  1904.       var needsRestart = items.length > 0;
  1905.       
  1906.       // Look for extensions that need to be enabled
  1907.       items = this._ds.getItemsWithFlagSet("toBeEnabled");
  1908.       for (var i = 0; i < items.length; ++i)
  1909.         this._finalizeEnableDisable(items[i], false);
  1910.       
  1911.       // Look for extensions that need to be disabled
  1912.       items = this._ds.getItemsWithFlagSet("toBeDisabled");
  1913.       for (var i = 0; i < items.length; ++i)
  1914.         this._finalizeEnableDisable(items[i], true);
  1915.       
  1916.       // Look for extensions that need to be removed. This MUST be done after
  1917.       // the install operations since extensions to be installed may have to be
  1918.       // uninstalled if there are errors during the installation process!
  1919.       items = this._ds.getItemsWithFlagSet("toBeUninstalled");
  1920.       for (var i = 0; i < items.length; ++i)
  1921.         this._finalizeUninstall(items[i]);
  1922.         
  1923.       // Clean up any helper objects
  1924.       delete this._extInstaller;
  1925.       delete this._extUninstaller;
  1926.       delete this._extEnabler;
  1927.       
  1928.       this._updateManifests();
  1929.       // If no additional restart is required, it implies that there are
  1930.       // no new components that need registering so we can inform the app
  1931.       // not to do any extra startup checking next time round.    
  1932.       this._writeCompatibilityManifest(needsRestart);
  1933.     }
  1934.     catch (e) {
  1935.       dump("*** nsExtensionManager::_finishOperations - failure, catching exception so finalize window can close " + e +"\n");
  1936.     }
  1937.     win.close();
  1938.     
  1939.     return needsRestart;
  1940.   },
  1941.   
  1942.   // XXXben - this is actually a cheap stunt to load all the chrome registry 
  1943.   //          services required to register/unregister packages... the synchronous
  1944.   //          nature of this code ensures the window will never actually appear
  1945.   //          on screen. 
  1946.   _showProgressWindow: function nsExtensionManager__showProgressWindow ()
  1947.   {
  1948.     var ww = Components.classes["@mozilla.org/embedcomp/window-watcher;1"]
  1949.                        .getService(Components.interfaces.nsIWindowWatcher);
  1950.     return ww.openWindow(null, "chrome://mozapps/content/extensions/finalize.xul", 
  1951.                          "", "chrome,centerscreen,dialog", null);
  1952.   },
  1953.   
  1954.   _loadDefaults: function nsExtensionManager__loadDefaults ()
  1955.   {
  1956.     // Load default preferences files for all extensions
  1957.     var defaultsManifest = getFile(KEY_PROFILEDIR, 
  1958.                                    [DIR_EXTENSIONS, FILE_DEFAULTS]);
  1959.     if (defaultsManifest.exists()) {
  1960.       var fis = Components.classes["@mozilla.org/network/file-input-stream;1"]
  1961.                           .createInstance(Components.interfaces.nsIFileInputStream);
  1962.       fis.init(defaultsManifest, -1, -1, false);
  1963.       var lis = fis.QueryInterface(Components.interfaces.nsILineInputStream);
  1964.       var line = { value: "" };
  1965.       var more = false;
  1966.       do {
  1967.         more = lis.readLine(line);
  1968.         var lf = Components.classes["@mozilla.org/file/local;1"]
  1969.                             .createInstance(Components.interfaces.nsILocalFile);
  1970.         var path = line.value;
  1971.         if (path) {
  1972.           lf.initWithPath(path);
  1973.           
  1974.           if (lf.exists())
  1975.             gPref.readUserPrefs(lf);
  1976.         }
  1977.       }
  1978.       while (more);
  1979.       fis.close();
  1980.     }
  1981.   },
  1982.   
  1983.   ensurePreConfiguredItem: function nsExtensionManager_ensurePreConfiguredItem (aItemID, aItemType, aManifest)
  1984.   {
  1985.     this._ds.insertForthcomingItem(aItemID, aItemType, false);
  1986.     var metadataDS = getInstallManifest(aManifest);
  1987.     this._ds.addItemMetadata(aItemID, aItemType, metadataDS, false);
  1988.   },
  1989.   
  1990.   checkForMismatches: function nsExtensionManager_checkForMismatches () 
  1991.   {
  1992.     var needsRestart = false;
  1993.         
  1994.     this._disableObsoleteExtensions();
  1995.  
  1996.     // Check to see if the version of the application that is being started
  1997.     // now is the same one that was started last time. 
  1998.     var currAppVersion = gPref.getCharPref(PREF_EM_APP_EXTENSIONS_VERSION);
  1999.     try {
  2000.       var lastAppVersion = gPref.getCharPref(PREF_EM_LAST_APP_VERSION);
  2001.     }
  2002.     catch (e) {}
  2003.     if (currAppVersion != lastAppVersion) {
  2004.       // Version mismatch, we're have to load the extensions datasource
  2005.       // and do version checking. Time hit here doesn't matter since this 
  2006.       // doesn't happen all that often.
  2007.       this._ensureDS();
  2008.       var currAppID = gPref.getCharPref(PREF_EM_APP_ID);
  2009.       var items = this._ds.getIncompatibleItemList(currAppID, currAppVersion,
  2010.                                                    nsIUpdateItem.TYPE_ADDON);
  2011.       if (items.length > 0) {
  2012.         for (var i = 0; i < items.length; ++i) {
  2013.           // Now disable the extension so it won't hurt anything. 
  2014.           var itemType = getItemType(this._ds._getResourceForItem(items[i].id).Value);
  2015.           if (itemType != -1 && itemType & nsIUpdateItem.TYPE_EXTENSION)
  2016.             this.disableExtension(items[i].id);
  2017.           else if (itemType & nsIUpdateItem.TYPE_THEME) {
  2018.             if (gPref.prefHasUserValue(PREF_GENERAL_SKINS_SELECTEDSKIN))
  2019.               gPref.clearUserPref(PREF_GENERAL_SKINS_SELECTEDSKIN);
  2020.             this._ds.disableTheme(items[i].id);
  2021.           }
  2022.         }
  2023.         var updates = Components.classes["@mozilla.org/updates/update-service;1"]
  2024.                                 .getService(Components.interfaces.nsIUpdateService);
  2025.         updates.checkForUpdates(items, items.length, nsIUpdateItem.TYPE_ADDON, 
  2026.                                 nsIUpdateService.SOURCE_EVENT_MISMATCH,
  2027.                                 null);
  2028.         
  2029.         needsRestart = true;
  2030.       }
  2031.     }
  2032.     
  2033.     // Somehow the component list went away, and for that reason the new one
  2034.     // generated by this function is going to result in a different compreg.
  2035.     // We must force a restart.
  2036.     var componentList = getFile(KEY_PROFILEDIR, [FILE_COMPONENT_MANIFEST]);
  2037.     if (!componentList.exists())
  2038.       needsRestart = true;
  2039.     
  2040.     // Now update the last app version so we don't do this checking 
  2041.     // again. 
  2042.     gPref.setCharPref(PREF_EM_LAST_APP_VERSION, currAppVersion);
  2043.  
  2044.     // XXXben - I am not entirely sure this is needed, since components and 
  2045.     // defaults manifests are written by the disabling function. Not going to
  2046.     // rock the boat now however. 
  2047.     this._updateManifests();
  2048.     
  2049.     return needsRestart;
  2050.   },
  2051.   
  2052.   get inSafeMode() 
  2053.   {
  2054.     return this._ds.safeMode;
  2055.   },
  2056.   
  2057.   _updateManifests: function nsExtensionManager__updateManifests ()
  2058.   {
  2059.     // Update the components manifests with paths for compatible, enabled, 
  2060.     // extensions.
  2061.     try {
  2062.       // Wrap this in try..catch so that if the account is restricted we don't
  2063.       // completely fail here for lack of permissions to write to the bin
  2064.       // dir (and cause apprunner to go into a restart loop). 
  2065.       //
  2066.       // This means that making changes to install-dir extensions only possible
  2067.       // for people with write access to bin dir (i.e. uninstall, disable, 
  2068.       // enable)
  2069.       this._writeComponentManifest(false);
  2070.       this._writeDefaults(false);
  2071.     }
  2072.     catch (e) { 
  2073.       dump("*** ExtensionManager:_updateManifests: no access privileges to application directory, skipping.\n"); 
  2074.     };
  2075.     this._writeComponentManifest(true);
  2076.     this._writeDefaults(true);
  2077.   },
  2078.   
  2079.   // XXXben write to temporary file then move to final when done.
  2080.   _writeProfileFile: function nsExtensionManager__writeProfileFile (aFile, aGetDirFunc, aIsProfile)
  2081.   {
  2082.     // When an operation is performed that requires a component re-registration
  2083.     // (extension enabled/disabled, installed, uninstalled), we must write the
  2084.     // set of registry-relative paths of components to register to an .autoreg 
  2085.     // file which lives in the profile folder. 
  2086.     //
  2087.     // To do this we must enumerate all installed extensions and write data 
  2088.     // about all valid items to the file. 
  2089.     this._ensureDS();
  2090.     
  2091.     var fos = Components.classes["@mozilla.org/network/file-output-stream;1"]
  2092.                         .createInstance(Components.interfaces.nsIFileOutputStream);
  2093.     const MODE_WRONLY   = 0x02;
  2094.     const MODE_CREATE   = 0x08;
  2095.     const MODE_TRUNCATE = 0x20;
  2096.     fos.init(aFile, MODE_WRONLY | MODE_CREATE | MODE_TRUNCATE, 0644, 0);
  2097.  
  2098.     var extensions = this.getItemList(null, nsIUpdateItem.TYPE_EXTENSION, { });
  2099.     var validExtensions = [];
  2100.     for (var i = 0; i < extensions.length; ++i) {
  2101.       var extension = extensions[i];
  2102.     
  2103.       // An extension entry is valid only if it is not disabled, not about to 
  2104.       // be disabled, and not about to be uninstalled.
  2105.       var toBeDisabled = this._ds.getItemProperty(extension.id, "toBeDisabled");
  2106.       var toBeUninstalled = this._ds.getItemProperty(extension.id, "toBeUninstalled");
  2107.       var toBeInstalled = this._ds.getItemProperty(extension.id, "toBeInstalled");
  2108.       var disabled = this._ds.getItemProperty(extension.id, "disabled");
  2109.       if (toBeDisabled == "true" || toBeUninstalled == "true" || 
  2110.           disabled == "true" || toBeInstalled == "true")
  2111.         continue;
  2112.       
  2113.       var isProfile = this._ds.isProfileItem(extension.id);
  2114.       var sourceDir = aGetDirFunc(isProfile, extension.id);
  2115.       if (sourceDir.exists() && (aIsProfile == isProfile))
  2116.         validExtensions.push({ sourceDir: sourceDir, isProfile: isProfile });
  2117.     }
  2118.     
  2119.     var lines = ["[Extra Files]\r\n",
  2120.                  "Count=" + validExtensions.length + "\r\n"];
  2121.     for (i = 0; i < lines.length; ++i)
  2122.       fos.write(lines[i], lines[i].length);
  2123.       
  2124.     for (i = 0; i < validExtensions.length; ++i) {
  2125.       var e = validExtensions[i];
  2126.       var relativeDir = getDir(e.isProfile ? KEY_PROFILEDIR : KEY_APPDIR, []);
  2127.       var lf = e.sourceDir.QueryInterface(Components.interfaces.nsILocalFile);
  2128.       var relDesc = lf.getRelativeDescriptor(relativeDir);
  2129.       var line = "File" + i + "=" + relDesc + "\r\n";
  2130.       fos.write(line, line.length);
  2131.     }
  2132.     fos.close();
  2133.   },
  2134.   
  2135.   _getComponentsDir: function nsExtensionManager__getComponentsDir (aIsProfile, aExtensionID)
  2136.   {
  2137.     return getDirNoCreate(getDirKey(aIsProfile), 
  2138.                           [DIR_EXTENSIONS, aExtensionID, DIR_COMPONENTS]);
  2139.   },
  2140.  
  2141.   _getPreferencesDir: function nsExtensionManager__getPreferencesDir (aIsProfile, aExtensionID)
  2142.   {
  2143.     return getDirNoCreate(getDirKey(aIsProfile), 
  2144.                           [DIR_EXTENSIONS, aExtensionID, 
  2145.                            DIR_DEFAULTS, DIR_DEFAULTS_PREFS]);
  2146.   },
  2147.  
  2148.   _writeComponentManifest: function nsExtensionManager__writeComponentManifest (aIsProfile)
  2149.   {
  2150.     var manifest = aIsProfile ? getFile(KEY_PROFILEDIR, [FILE_COMPONENT_MANIFEST]) : 
  2151.                                 getFile(KEY_APPDIR, [FILE_COMPONENT_MANIFEST]);
  2152.     this._writeProfileFile(manifest, this._getComponentsDir, aIsProfile);
  2153.  
  2154.     // Now refresh the compatibility manifest.
  2155.     this._writeCompatibilityManifest(true);
  2156.   },
  2157.   
  2158.   _writeCompatibilityManifest: function nsExtensionManager__writeCompatibilityManifest (aComponentListUpdated)
  2159.   {
  2160.     var fos = Components.classes["@mozilla.org/network/file-output-stream;1"]
  2161.                         .createInstance(Components.interfaces.nsIFileOutputStream);
  2162.     const MODE_WRONLY   = 0x02;
  2163.     const MODE_CREATE   = 0x08;
  2164.     const MODE_TRUNCATE = 0x20;
  2165.  
  2166.     // The compat file only lives in the Profile dir because we make the 
  2167.     // assumption that you can never have extensions prior to profile
  2168.     // startup.
  2169.     var compat = getFile(KEY_PROFILEDIR, [FILE_COMPAT_MANIFEST]);
  2170.     fos.init(compat, MODE_WRONLY | MODE_CREATE | MODE_TRUNCATE, 0644, 0);
  2171.  
  2172.     var currAppBuildID = gPref.getCharPref(PREF_EM_APP_BUILDID);
  2173.  
  2174.     var val = aComponentListUpdated ? 1 : 0;
  2175.     var lines = ["[Compatibility]\r\n",
  2176.                  "Build ID=" + currAppBuildID + "\r\n",
  2177.                  "Components List Changed=" + val + "\r\n"];
  2178.     for (var i = 0; i < lines.length; ++i)
  2179.       fos.write(lines[i], lines[i].length);
  2180.  
  2181.     fos.close();
  2182.   },
  2183.   
  2184.   _writeDefaults: function nsExtensionManager__writeDefaults (aIsProfile)
  2185.   {
  2186.     var manifest = aIsProfile ? getFile(KEY_PROFILEDIR, [FILE_DEFAULTS]) : 
  2187.                                 getFile(KEY_APPDIR, [FILE_DEFAULTS]);
  2188.     this._writeProfileFile(manifest, this._getPreferencesDir, aIsProfile);
  2189.   },
  2190.   
  2191.   _cleanDirs: function nsExtensionManager__cleanDirs ()
  2192.   {
  2193.     var keys = [KEY_PROFILEDIR, KEY_APPDIR];
  2194.     for (var i = 0; i < keys.length; ++i) {
  2195.       var extensions = getDir(keys[i], [DIR_EXTENSIONS]);
  2196.       var entries = extensions.directoryEntries;
  2197.       while (entries.hasMoreElements()) {
  2198.         var entry = entries.getNext().QueryInterface(Components.interfaces.nsIFile);
  2199.         if (entry.isDirectory() && !entry.directoryEntries.hasMoreElements()) {
  2200.           try {
  2201.             entry.remove(false);
  2202.           }
  2203.           catch (e) { }
  2204.         }
  2205.       }
  2206.     }
  2207.   },
  2208.   
  2209.   /////////////////////////////////////////////////////////////////////////////  
  2210.   // nsIExtensionManager
  2211.   installExtension: function nsExtensionManager_installExtension (aXPIFile, aFlags)
  2212.   {
  2213.     // Since we're installing a "new type" extension, we assume a file layout
  2214.     // within the XPI like so:
  2215.     // foo.xpi/
  2216.     //         extension.rdf
  2217.     //         chrome/
  2218.     //         components/ 
  2219.     //         defaults/
  2220.     //                  prefs/
  2221.     var installProfile = aFlags & nsIExtensionManager.FLAG_INSTALL_PROFILE;
  2222.  
  2223.     var tempDir = getDir(getDirKey(installProfile), [DIR_EXTENSIONS, DIR_TEMP]);
  2224.     var fileName = getRandomFileName("temp", "xpi");
  2225.     aXPIFile.copyTo(tempDir, fileName);
  2226.     var xpiFile = tempDir.clone();
  2227.     xpiFile.append(fileName);
  2228.  
  2229.     // if the source file was read-only, fix permissions
  2230.     if (!xpiFile.isWritable()) {
  2231.       xpiFile.permissions = 0644;
  2232.     }
  2233.  
  2234.     var zipReader = Components.classes["@mozilla.org/libjar/zip-reader;1"]
  2235.                               .createInstance(Components.interfaces.nsIZipReader);
  2236.     zipReader.init(xpiFile);
  2237.     zipReader.open();
  2238.     
  2239.     var tempManifest = getFile(getDirKey(installProfile),
  2240.                                [DIR_EXTENSIONS, DIR_TEMP, getRandomFileName("install", "rdf")]);
  2241.     if (!tempManifest.exists())
  2242.       tempManifest.create(Components.interfaces.nsIFile.NORMAL_FILE_TYPE, 0644);
  2243.     zipReader.extract(FILE_INSTALL_MANIFEST, tempManifest);
  2244.     
  2245.     var extensionID = this.installExtensionInternal(xpiFile, tempManifest, installProfile);
  2246.     switch (extensionID) {
  2247.     case ERROR_EXTENSION_IS_THEME:
  2248.       this.installTheme(aXPIFile, aFlags);
  2249.       break;
  2250.     case ERROR_INVALID_VERSION:
  2251.     case ERROR_PHONED_HOME:
  2252.       break;
  2253.     default:
  2254.       // Then we stage the extension's XPI into a temporary directory so we 
  2255.       // can extract them after the next restart. 
  2256.       this._stageExtensionXPI(zipReader, extensionID, installProfile);
  2257.  
  2258.       this._writeComponentManifest(installProfile);
  2259.     }
  2260.     
  2261.     zipReader.close();
  2262.     tempManifest.remove(false);
  2263.     
  2264.     if (extensionID != ERROR_PHONED_HOME)
  2265.       xpiFile.remove(false);
  2266.   },
  2267.   
  2268.   installExtensionInternal: function nsExtensionManager_installExtensionInternal (aXPIFile, aManifest, aIsProfile)
  2269.   {
  2270.     var ds = getInstallManifest(aManifest);
  2271.     if (!ds) return;
  2272.     
  2273.     // XXXben - this is a hack until we properly fix xpinstall to be able to install
  2274.     //          different chrome types from trusted script. At the moment, when we
  2275.     //          call initManagerFromChrome, we can only install extensions, since
  2276.     //          the code path that installs themes is not utilized. To minimize the
  2277.     //          level of changes happening at the lower level in xpinstall at this
  2278.     //          point I am inserting this hack which checks for a theme-only property
  2279.     //          in the install manifest.
  2280.     var manifestRoot = gRDF.GetResource("urn:mozilla:install-manifest");
  2281.     var internalName = gRDF.GetResource(EM_NS("internalName"));
  2282.     if (stringData(ds.GetTarget(manifestRoot, internalName, true)) != "--")
  2283.       return ERROR_EXTENSION_IS_THEME;
  2284.     
  2285.     // We do a basic version check first just to make sure we somehow weren't 
  2286.     // tricked into installing an incompatible extension...
  2287.     this._ensureDS();
  2288.     var extensionID = this.canInstallItem(ds);
  2289.     // |extensionID| must be a GUID string, not a number - a number means failure.
  2290.     if (isNaN(parseInt(extensionID)))
  2291.       this._configureForthcomingItem(ds, extensionID, aIsProfile);
  2292.     else if (extensionID == 0) {
  2293.       var io = new this.IncompatibleObserver(this);
  2294.       var isChecking = io.checkForUpdates(ds, nsIUpdateItem.TYPE_EXTENSION,
  2295.                                           aXPIFile, aIsProfile);
  2296.       if (!isChecking)
  2297.         showIncompatibleError(ds);
  2298.       else {
  2299.         extensionID = ERROR_PHONED_HOME; // caller uses this to distinguish 
  2300.                                          // phone-home attempt.
  2301.       }
  2302.     }
  2303.     
  2304.     return extensionID;
  2305.   },
  2306.   
  2307.   IncompatibleObserver: function nsExtensionManager_IncompatibleObserver (aEM) 
  2308.   {
  2309.     this._item = null;
  2310.     this._em = aEM;
  2311.     this._ds = null;
  2312.     this._xpi = null;
  2313.     this._extensionID = 0;
  2314.     this._isProfile = true;
  2315.     
  2316.     this.checkForUpdates = function nsExtensionManager__iOcheckForUpdates (aDataSource, aType, 
  2317.                                                                            aXPIFile, aIsProfile)
  2318.     {
  2319.       // Construct a nsIUpdateItem for this extension...
  2320.       var item = this._em._getItemForIncompatibleID(aDataSource, aType);
  2321.       if (item) {
  2322.         this._item      = item;
  2323.         this._ds        = aDataSource;
  2324.         this._xpi       = aXPIFile;
  2325.         this._isProfile = true;
  2326.         
  2327.         gOS.addObserver(this, "Update:Extension:Started", false);
  2328.         gOS.addObserver(this, "Update:Extension:Item-Ended", false);
  2329.         gOS.addObserver(this, "Update:Extension:Item-Error", false);
  2330.         gOS.addObserver(this, "Update:Extension:Ended", false);
  2331.  
  2332.         this._em.update([item], 1, true);
  2333.         
  2334.         return true;
  2335.       }
  2336.       return false;
  2337.     }
  2338.     
  2339.     this.observe = function nsExtensionManager__iOobserve (aSubject, aTopic, aData)
  2340.     {
  2341.       switch (aTopic) {
  2342.       case "Update:Extension:Started":
  2343.         break;
  2344.       case "Update:Extension:Item-Ended":
  2345.         if (aSubject) {
  2346.           var item = aSubject.QueryInterface(Components.interfaces.nsIUpdateItem);
  2347.           this._em._ds.setTargetApplicationInfo(item.id, 
  2348.                                                 item.minAppVersion,
  2349.                                                 item.maxAppVersion, 
  2350.                                                 this._ds, 
  2351.                                                 this._item.type);
  2352.           this._extensionID = this._em.canInstallItem(this._ds);
  2353.         }
  2354.         break;
  2355.       case "Update:Extension:Item-Error":
  2356.         break;
  2357.       case "Update:Extension:Ended":
  2358.         gOS.removeObserver(this, "Update:Extension:Started");
  2359.         gOS.removeObserver(this, "Update:Extension:Item-Ended");
  2360.         gOS.removeObserver(this, "Update:Extension:Item-Error");
  2361.         gOS.removeObserver(this, "Update:Extension:Ended");
  2362.         
  2363.         if (isNaN(this._extensionID)) {
  2364.           var zipReader = Components.classes["@mozilla.org/libjar/zip-reader;1"]
  2365.                                     .createInstance(Components.interfaces.nsIZipReader);
  2366.           zipReader.init(this._xpi);
  2367.           zipReader.open();
  2368.  
  2369.           // Add the item after all
  2370.           this._em._configureForthcomingItem(this._ds, this._extensionID, 
  2371.                                              this._isProfile);
  2372.           this._em._stageExtensionXPI(zipReader, this._extensionID, this._isProfile);
  2373.           this._em._writeComponentManifest(this._isProfile);
  2374.           
  2375.           zipReader.close();
  2376.         }
  2377.         else 
  2378.           showIncompatibleError(this._ds);
  2379.         
  2380.         // Now really delete the temporary XPI file
  2381.         this._xpi.remove(false);
  2382.         break;
  2383.       }
  2384.     }
  2385.   },
  2386.   
  2387.   _configureForthcomingItem: function nsExtensionManager__configureForthcomingItem (aDataSource, 
  2388.                                                                                     aExtensionID, 
  2389.                                                                                     aIsProfile)
  2390.   {
  2391.     // Clear any "disabled" flags that may have been set by the mismatch 
  2392.     // checking code at startup.
  2393.     var props = { toBeDisabled  : null,
  2394.                   disabled      : null,
  2395.                   toBeInstalled : this._ds._emL("true"),
  2396.                   name          : this.getManifestProperty(aDataSource, "name"),
  2397.                   version       : this.getManifestProperty(aDataSource, "version") };
  2398.     for (var p in props) {
  2399.       this._ds.setItemProperty(aExtensionID, this._ds._emR(p),
  2400.                                props[p], aIsProfile,
  2401.                                nsIUpdateItem.TYPE_EXTENSION);
  2402.     }
  2403.  
  2404.     // Insert it into the child list NOW rather than later because:
  2405.     // - extensions installed using the command line need to be a member
  2406.     //   of a container during the install phase for the code to be able
  2407.     //   to identify profile vs. global
  2408.     // - extensions installed through the UI should show some kind of
  2409.     //   feedback to indicate their presence is forthcoming (i.e. they
  2410.     //   will be available after a restart).
  2411.     this._ds.insertForthcomingItem(aExtensionID, nsIUpdateItem.TYPE_EXTENSION, 
  2412.                                    aIsProfile);
  2413.   },
  2414.   
  2415.   _getItemForIncompatibleID: function nsExtensionManager__getItemForID (aDataSource, aType)
  2416.   {
  2417.     var newItem = null;
  2418.     var id, version, targetAppInfo, name, updateURL;
  2419.     var manifestRoot = gRDF.GetResource("urn:mozilla:install-manifest");
  2420.     try {
  2421.       function getProperty (aDataSource, aSourceResource, aProperty)
  2422.       {
  2423.         var rv;
  2424.         try {
  2425.           var property = gRDF.GetResource(EM_NS(aProperty));
  2426.           rv = stringData(aDataSource.GetTarget(aSourceResource, property, true));
  2427.           if (rv == "--")
  2428.             throw Components.results.NS_ERROR_FAILURE;
  2429.         }
  2430.         catch (e) { }
  2431.         return rv;
  2432.       }
  2433.       
  2434.       var root = gRDF.GetResource("urn:mozilla:install-manifest");
  2435.       id            = getProperty(aDataSource, root, "id");
  2436.       version       = getProperty(aDataSource, root, "version");
  2437.       targetAppInfo = this._ds.getTargetApplicationInfo(id, aDataSource, aType);
  2438.       name          = getProperty(aDataSource, root, "name");
  2439.       updateURL     = getProperty(aDataSource, root, "updateURL");
  2440.       if (updateURL == "--")
  2441.         updateURL = "";
  2442.       
  2443.       newItem = Components.classes["@mozilla.org/updates/item;1"]
  2444.                           .createInstance(Components.interfaces.nsIUpdateItem);
  2445.       newItem.init(id, version, targetAppInfo.minVersion, 
  2446.                    targetAppInfo.maxVersion,
  2447.                    name, -1, "", "", updateURL, aType);
  2448.     }
  2449.     catch (e) {
  2450.       return null;
  2451.     }
  2452.     return newItem;
  2453.   },
  2454.   
  2455.   canInstallItem: function nsExtensionManager_canInstallItem (aDataSource)
  2456.   {
  2457.     var manifestRoot = gRDF.GetResource("urn:mozilla:install-manifest");
  2458.     // First make sure the item has a valid "version" property. 
  2459.     var version = gRDF.GetResource(EM_NS("version"));
  2460.     var versionLiteral = stringData(aDataSource.GetTarget(manifestRoot, version, true));
  2461.     if (!getVersionChecker().isValidVersion(versionLiteral)) {
  2462.       var name = gRDF.GetResource(EM_NS("name"));
  2463.       var nameLiteral = stringData(aDataSource.GetTarget(manifestRoot, name, true));
  2464.       showInvalidVersionError(nameLiteral, versionLiteral);
  2465.       return ERROR_INVALID_VERSION;
  2466.     }
  2467.         
  2468.     // Check the target application range specified by the extension metadata.
  2469.     if (this._ds.isCompatible(aDataSource, manifestRoot)) {
  2470.       var id = gRDF.GetResource(EM_NS("id"));
  2471.       var idLiteral = aDataSource.GetTarget(manifestRoot, id, true);
  2472.       return idLiteral.QueryInterface(Components.interfaces.nsIRDFLiteral).Value;
  2473.     }
  2474.     return 0;
  2475.   },
  2476.   
  2477.   getManifestProperty: function nsExtensionManager_getManifestProperty (aDataSource, aProperty)
  2478.   {
  2479.     var manifestRoot = gRDF.GetResource("urn:mozilla:install-manifest");
  2480.     var arc = gRDF.GetResource(EM_NS(aProperty));
  2481.     return aDataSource.GetTarget(manifestRoot, arc, true);
  2482.   },
  2483.   
  2484.   _stageExtensionXPI: function nsExtensionManager__stageExtensionXPI (aZipReader, aExtensionID, aInstallProfile)
  2485.   {
  2486.     // Get the staging dir
  2487.     var dir = getDir(getDirKey(aInstallProfile),
  2488.                      [DIR_EXTENSIONS, DIR_TEMP, aExtensionID]);
  2489.     var extensionFileName = aExtensionID + ".xpi";
  2490.     var extensionFile = dir.clone();
  2491.     extensionFile.append(extensionFileName);
  2492.     if (extensionFile.exists())
  2493.       extensionFile.remove(false);
  2494.     aZipReader.file.copyTo(dir, extensionFileName);
  2495.  
  2496.     // if the source file was readonly, fix the permissions
  2497.     if (!extensionFile.isWritable()) {
  2498.       extensionFile.permissions = 0644;
  2499.     }
  2500.   },
  2501.   
  2502.   // This function is called on the next startup 
  2503.   _finalizeInstall: function nsExtensionManager__finalizeInstall (aExtensionID)
  2504.   {
  2505.     var isProfile = this._ds.isProfileItem(aExtensionID);
  2506.     if (aExtensionID == 0 || aExtensionID == -1) {
  2507.       this._ds.removeCorruptItem(aExtensionID, 
  2508.                                  nsIUpdateItem.TYPE_EXTENSION, 
  2509.                                  isProfile);
  2510.       return;
  2511.     }
  2512.     
  2513.     if (!this._extInstaller)
  2514.       this._extInstaller = new nsExtensionInstaller(this._ds);
  2515.       
  2516.     this._extInstaller.install(aExtensionID, isProfile);
  2517.     
  2518.     // Update the Components Manifest
  2519.     this._writeComponentManifest(isProfile);
  2520.     
  2521.     // Update the Defaults Manifest
  2522.     this._writeDefaults(isProfile);
  2523.   },
  2524.   
  2525.   _finalizeEnableDisable: function nsExtensionManager__finalizeEnableDisable (aExtensionID, aDisable)
  2526.   {
  2527.     if (!this._extEnabler)
  2528.       this._extEnabler = new nsExtensionEnabler(this._ds);
  2529.       
  2530.     var isProfile = this._ds.isProfileItem(aExtensionID);
  2531.     this._extEnabler.enable(aExtensionID, isProfile, aDisable);
  2532.  
  2533.     // clear temporary flags
  2534.     this._ds.setItemProperty(aExtensionID, 
  2535.                              this._ds._emR("toBeEnabled"),
  2536.                              null, isProfile,
  2537.                              nsIUpdateItem.TYPE_EXTENSION);
  2538.     this._ds.setItemProperty(aExtensionID, 
  2539.                              this._ds._emR("toBeDisabled"),
  2540.                              null, isProfile,
  2541.                              nsIUpdateItem.TYPE_EXTENSION);
  2542.   },
  2543.   
  2544.   _finalizeUninstall: function nsExtensionManager__finalizeUninstall (aExtensionID)
  2545.   {
  2546.     if (!this._extUninstaller)
  2547.       this._extUninstaller = new nsExtensionUninstaller(this._ds);
  2548.     var isProfile = this._ds.isProfileItem(aExtensionID);
  2549.     this._extUninstaller.uninstall(aExtensionID, isProfile);
  2550.  
  2551.     // Clean the extension resource
  2552.     this._ds.removeItemMetadata(aExtensionID, nsIUpdateItem.TYPE_EXTENSION);
  2553.     
  2554.     // Do this LAST since inferences are made about an item based on
  2555.     // what container it's in.
  2556.     this._ds.removeItemFromContainer(aExtensionID, 
  2557.                                      nsIUpdateItem.TYPE_EXTENSION,
  2558.                                      isProfile);
  2559.   },
  2560.       
  2561.   uninstallExtension: function nsExtensionManager_uninstallExtension (aExtensionID)
  2562.   {
  2563.     if (!this._ds.isDownloadItem(aExtensionID)) {
  2564.       this._ds.uninstallExtension(aExtensionID);
  2565.  
  2566.       var isProfile = this._ds.isProfileItem(aExtensionID);
  2567.  
  2568.       // Update the Components Manifest
  2569.       this._writeComponentManifest(isProfile);
  2570.  
  2571.       // Update the Defaults Manifest
  2572.       this._writeDefaults(isProfile);
  2573.     }
  2574.     else {
  2575.       // Bad download entry - uri is url, e.g. "http://www.foo.com/test.xpi"
  2576.       // ... just remove it from the list. 
  2577.       this._ds.removeCorruptDLItem(aExtensionID, nsIUpdateItem.TYPE_EXTENSION);
  2578.     }
  2579.   },
  2580.   
  2581.   enableExtension: function nsExtensionManager_enableExtension (aExtensionID)
  2582.   {
  2583.     this._ds.enableExtension(aExtensionID);
  2584.  
  2585.     var isProfile = this._ds.isProfileItem(aExtensionID);
  2586.  
  2587.     // Update the Components Manifest
  2588.     this._writeComponentManifest(isProfile);
  2589.  
  2590.     // Update the Defaults Manifest
  2591.     this._writeDefaults(isProfile);
  2592.   },
  2593.   
  2594.   disableExtension: function nsExtensionManager_disableExtension (aExtensionID)
  2595.   {
  2596.     this._ds.disableExtension(aExtensionID);
  2597.  
  2598.     var isProfile = this._ds.isProfileItem(aExtensionID);
  2599.  
  2600.     // Update the Components Manifest
  2601.     this._writeComponentManifest(isProfile);
  2602.  
  2603.     // Update the Defaults Manifest
  2604.     this._writeDefaults(isProfile);
  2605.   },
  2606.   
  2607.   enableTheme: function nsExtensionsDataSource_enableTheme (aThemeID)
  2608.   {
  2609.     this._ds.enableTheme(aThemeID);
  2610.   },
  2611.     
  2612.   disableTheme: function nsExtensionsDataSource_disableTheme (aThemeID)
  2613.   {
  2614.     this._ds.disableTheme(aThemeID);
  2615.   },
  2616.   
  2617.   update: function nsExtensionManager_update (aItems, aItemCount, aVersionUpdateOnly)
  2618.   {
  2619.     var appID = gPref.getCharPref(PREF_EM_APP_ID);
  2620.     var appVersion = gPref.getCharPref(PREF_EM_APP_EXTENSIONS_VERSION);
  2621.  
  2622.     if (aItems.length == 0) {
  2623.       var addonType = nsIUpdateItem.TYPE_ADDON;
  2624.       aItems = this.getItemList(null, addonType, { });
  2625.     }
  2626.     var updater = new nsExtensionItemUpdater(appID, appVersion, this);
  2627.     updater.checkForUpdates(aItems, aItems.length, aVersionUpdateOnly);
  2628.   },
  2629.   
  2630.   getItemList: function nsExtensionManager_getItemList (aItemID, aType, aCountRef)
  2631.   {
  2632.     this._ensureDS();
  2633.     return this._ds.getItemList(aItemID, aType, aCountRef);
  2634.   },    
  2635.  
  2636.   /////////////////////////////////////////////////////////////////////////////  
  2637.   // Themes
  2638.   installTheme: function nsExtensionManager_installTheme (aJARFile, aFlags)
  2639.   {
  2640.     this._ensureDS();
  2641.     
  2642.     var isProfile = aFlags & nsIExtensionManager.FLAG_INSTALL_PROFILE;
  2643.     var installer = new nsThemeInstaller(this._ds, this);
  2644.     installer.install(aJARFile, isProfile);
  2645.     // XPInstall selects the theme, if necessary.
  2646.   },
  2647.   
  2648.   uninstallTheme: function nsExtensionManager_uninstallTheme (aThemeID)
  2649.   {
  2650.     if (!this._ds.isDownloadItem(aThemeID)) {
  2651.       this._ensureDS();
  2652.       this._ds.uninstallTheme(aThemeID);
  2653.     }
  2654.     else {
  2655.       // Bad download entry - uri is url, e.g. "http://www.foo.com/test.jar"
  2656.       // ... just remove it from the list. 
  2657.       this._ds.removeCorruptDLItem(aThemeID, nsIUpdateItem.TYPE_THEME);
  2658.     }
  2659.   },
  2660.   
  2661.   moveTop: function nsExtensionManager_moveTop (aItemID)
  2662.   {
  2663.     this._ds.moveTop(aItemID);
  2664.   },
  2665.   
  2666.   moveUp: function nsExtensionManager_moveUp (aItemID)
  2667.   {
  2668.     this._ds.moveUp(aItemID);
  2669.   },
  2670.   
  2671.   moveDown: function nsExtensionManager_moveDown (aItemID)
  2672.   {
  2673.     this._ds.moveDown(aItemID);
  2674.   },
  2675.  
  2676.   get datasource()
  2677.   {
  2678.     this._ensureDS();
  2679.     return this._ds;
  2680.   },
  2681.   
  2682.   /////////////////////////////////////////////////////////////////////////////    
  2683.   // Downloads
  2684.   _transactions: [],
  2685.   _downloadCount: 0,
  2686.   addDownloads: function nsExtensionManager_addDownloads (aItems, aItemCount)
  2687.   {
  2688.     this._downloadCount += aItemCount;
  2689.     
  2690.     var txn = new nsItemDownloadTransaction(this);
  2691.     for (var i = 0; i < aItemCount; ++i) {
  2692.       var currItem = aItems[i];
  2693.       var txnID = Math.round(Math.random() * 100);
  2694.       txn.addDownload(currItem.name, currItem.xpiURL, currItem.iconURL, 
  2695.                       currItem.type, txnID);
  2696.       this._transactions.push(txn);
  2697.     }
  2698.  
  2699.     // Kick off the download process for this transaction
  2700.     gOS.addObserver(this, "offline-requested", false);
  2701.     gOS.addObserver(this, "quit-application-requested", false);
  2702.     gOS.notifyObservers(txn, "xpinstall-progress", "open");  
  2703.   },
  2704.   
  2705.   removeDownload: function nsExtensionManager_removeDownload (aURL, aType)
  2706.   {
  2707.     for (var i = 0; i < this._transactions.length; ++i) {
  2708.       if (this._transactions[i].containsURL(aURL)) {
  2709.         this._transactions[i].removeDownload(aURL, aType);
  2710.         return;
  2711.       }
  2712.     } 
  2713.   },
  2714.   
  2715.   _removeAllDownloads: function nsExtensionManager__removeAllDownloads ()
  2716.   {
  2717.     for (var i = 0; i < this._transactions.length; ++i)
  2718.       this._transactions[i].removeAllDownloads();
  2719.   },
  2720.   
  2721.   // The nsIXPIProgressDialog implementation in the download transaction object
  2722.   // forwards notifications through these methods which we then pass on to any
  2723.   // front end objects implementing nsIExtensionDownloadProgressListener that 
  2724.   // are listening. We maintain the master state of download operations HERE, 
  2725.   // not in the front end, because if the user closes the extension or theme 
  2726.   // managers during the downloads we need to maintain state and not terminate
  2727.   // the download/install process. 
  2728.   onStateChange: function nsExtensionManager_onStateChange (aTransaction, aURL, aState, aValue)
  2729.   {
  2730.     if (!(aURL in this._progressData)) 
  2731.       this._progressData[aURL] = { };
  2732.     this._progressData[aURL].state = aState;
  2733.     
  2734.     for (var i = 0; i < this._downloadObservers.length; ++i)
  2735.       this._downloadObservers[i].onStateChange(aURL, aState, aValue);
  2736.  
  2737.     const nsIXPIProgressDialog = Components.interfaces.nsIXPIProgressDialog;
  2738.     switch (aState) {
  2739.     case nsIXPIProgressDialog.INSTALL_DONE:
  2740.       --this._downloadCount;
  2741.       break;
  2742.     case nsIXPIProgressDialog.DIALOG_CLOSE:
  2743.       for (var i = 0; i < this._transactions.length; ++i) {
  2744.         if (this._transactions[i].id == aTransaction.id) {
  2745.           this._transactions.splice(i, 1);
  2746.           delete aTransaction;
  2747.           break;
  2748.         }
  2749.       }
  2750.       break;
  2751.     }
  2752.   },
  2753.   
  2754.   _progressData: { },
  2755.   onProgress: function nsExtensionManager_onProgress (aURL, aValue, aMaxValue)
  2756.   {
  2757.     for (var i = 0; i < this._downloadObservers.length; ++i)
  2758.       this._downloadObservers[i].onProgress(aURL, aValue, aMaxValue);
  2759.     
  2760.     if (!(aURL in this._progressData)) 
  2761.       this._progressData[aURL] = { };
  2762.     this._progressData[aURL].progress = Math.round((aValue / aMaxValue) * 100);
  2763.   },
  2764.  
  2765.   _downloadObservers: [],
  2766.   addDownloadObserver: function nsExtensionManager_addDownloadObserver (aXPIProgressDialog)
  2767.   {
  2768.     for (var i = 0; i < this._downloadObservers.length; ++i) {
  2769.       if (this._downloadObservers[i] == aXPIProgressDialog)
  2770.         return i;
  2771.     }
  2772.     this._downloadObservers.push(aXPIProgressDialog);
  2773.     return this._downloadObservers.length - 1;
  2774.   },
  2775.   
  2776.   removeDownloadObserverAt: function nsExtensionManager_removeDownloadObserverAt (aIndex)
  2777.   {
  2778.     this._downloadObservers.splice(aIndex, 1);
  2779.     if (this._downloadCount != 0)
  2780.       this._ds.flushProgressInfo(this._progressData);
  2781.   },
  2782.  
  2783.   //
  2784.   _ds: null,
  2785.  
  2786.   /////////////////////////////////////////////////////////////////////////////    
  2787.   // Other
  2788.   
  2789.   // This should NOT be called until after the window is shown! 
  2790.   _ensureDS: function nsExtensionManager__ensureDS ()
  2791.   {
  2792.     if (!this._ds) {
  2793.       dump("*** loading the extensions datasource\n");
  2794.       this._ds = new nsExtensionsDataSource();
  2795.       if (this._ds) {
  2796.         this._ds.loadExtensions(false);
  2797.         this._ds.loadExtensions(true);
  2798.       }
  2799.       
  2800.       // Ensure any pre-configured items are initialized.
  2801.       (new nsInstalledExtensionReader(this)).read();
  2802.     }
  2803.   },
  2804.  
  2805.   /////////////////////////////////////////////////////////////////////////////
  2806.   // nsIClassInfo
  2807.   getInterfaces: function nsExtensionManager_getInterfaces (aCount)
  2808.   {
  2809.     var interfaces = [Components.interfaces.nsIExtensionManager,
  2810.                       Components.interfaces.nsIXPIProgressDialog,
  2811.                       Components.interfaces.nsIObserver];
  2812.     aCount.value = interfaces.length;
  2813.     return interfaces;
  2814.   },
  2815.   
  2816.   getHelperForLanguage: function nsExtensionManager_getHelperForLanguage (aLanguage)
  2817.   {
  2818.     return null;
  2819.   },
  2820.   
  2821.   get contractID() 
  2822.   {
  2823.     return "@mozilla.org/extensions/manager;1";
  2824.   },
  2825.   
  2826.   get classDescription()
  2827.   {
  2828.     return "Extension Manager";
  2829.   },
  2830.   
  2831.   get classID() 
  2832.   {
  2833.     return Components.ID("{8A115FAA-7DCB-4e8f-979B-5F53472F51CF}");
  2834.   },
  2835.   
  2836.   get implementationLanguage()
  2837.   {
  2838.     return Components.interfaces.nsIProgrammingLanguage.JAVASCRIPT;
  2839.   },
  2840.   
  2841.   get flags()
  2842.   {
  2843.     return Components.interfaces.nsIClassInfo.SINGLETON;
  2844.   },
  2845.  
  2846.   /////////////////////////////////////////////////////////////////////////////
  2847.   // nsISupports
  2848.   QueryInterface: function nsExtensionManager_QueryInterface (aIID) 
  2849.   {
  2850.     if (!aIID.equals(Components.interfaces.nsIExtensionManager) &&
  2851.         !aIID.equals(Components.interfaces.nsIObserver) &&
  2852.         !aIID.equals(Components.interfaces.nsISupports))
  2853.       throw Components.results.NS_ERROR_NO_INTERFACE;
  2854.     return this;
  2855.   }
  2856. };
  2857.  
  2858. ///////////////////////////////////////////////////////////////////////////////
  2859. //
  2860. // nsItemDownloadTransaction
  2861. //
  2862. //   This object implements nsIXPIProgressDialog and represents a collection of
  2863. //   XPI/JAR download and install operations. There is one 
  2864. //   nsItemDownloadTransaction per back-end XPInstallManager object. We maintain
  2865. //   a collection of separate transaction objects because it's possible to have
  2866. //   multiple separate XPInstall download/install operations going on 
  2867. //   simultaneously, each with its own XPInstallManager instance. For instance
  2868. //   you could start downloading two extensions and then download a theme. Each
  2869. //   of these operations would open the appropriate FE and have to be able to
  2870. //   track each operation independently.
  2871. //
  2872. function nsItemDownloadTransaction(aManager)
  2873. {
  2874.   this._manager = aManager;
  2875.   this._downloads = [];
  2876. }
  2877.  
  2878. nsItemDownloadTransaction.prototype = {
  2879.   _manager    : null,
  2880.   _downloads  : [],
  2881.   id          : -1,
  2882.   
  2883.   addDownload: function nsItemDownloadTransaction_addDownload (aName, aURL, aIconURL, aItemType, aID)
  2884.   {
  2885.     this._downloads.push({ url: aURL, type: aItemType, waiting: true });
  2886.     this._manager._ds.addDownload(aName, aURL, aIconURL, aItemType);
  2887.     this.id = aID;
  2888.   },
  2889.   
  2890.   removeDownload: function nsItemDownloadTransaction_removeDownload (aURL, aItemType)
  2891.   {
  2892.     this._manager._ds.removeDownload(aURL, aItemType);
  2893.   },
  2894.   
  2895.   removeAllDownloads: function nsItemDownloadTransaction_removeAllDownloads ()
  2896.   {
  2897.     for (var i = 0; i < this._downloads.length; ++i)
  2898.       this.removeDownload(this._downloads[i].url, this._downloads[i].type);
  2899.   },
  2900.   
  2901.   containsURL: function nsItemDownloadTransaction_containsURL (aURL)
  2902.   {
  2903.     for (var i = 0; i < this._downloads.length; ++i) {
  2904.       if (this._downloads[i].url == aURL)
  2905.         return true;
  2906.     }
  2907.     return false;
  2908.   },
  2909.  
  2910.   /////////////////////////////////////////////////////////////////////////////  
  2911.   // nsIXPIProgressDialog
  2912.   onStateChange: function nsItemDownloadTransaction_onStateChange (aIndex, aState, aValue)
  2913.   {
  2914.     this._manager.onStateChange(this, this._downloads[aIndex].url, aState, aValue);
  2915.   },
  2916.   
  2917.   onProgress: function nsItemDownloadTransaction_onProgress (aIndex, aValue, aMaxValue)
  2918.   {
  2919.     this._manager.onProgress(this._downloads[aIndex].url, aValue, aMaxValue);
  2920.   },
  2921.   
  2922.   /////////////////////////////////////////////////////////////////////////////
  2923.   // nsISupports
  2924.   QueryInterface: function nsItemDownloadTransaction_QueryInterface (aIID) 
  2925.   {
  2926.     if (!aIID.equals(Components.interfaces.nsIXPIProgressDialog) &&
  2927.         !aIID.equals(Components.interfaces.nsISupports))
  2928.       throw Components.results.NS_ERROR_NO_INTERFACE;
  2929.     return this;
  2930.   }
  2931. };
  2932.  
  2933. ///////////////////////////////////////////////////////////////////////////////
  2934. //
  2935. // nsExtensionItemUpdater
  2936. //
  2937. function nsExtensionItemUpdater(aTargetAppID, aTargetAppVersion, aEM) 
  2938. {
  2939.   this._appID = aTargetAppID;
  2940.   this._appVersion = aTargetAppVersion;
  2941.   this._emDS = aEM._ds;
  2942.   this._em = aEM;
  2943.  
  2944.   getVersionChecker();
  2945. }
  2946.  
  2947. nsExtensionItemUpdater.prototype = {
  2948.   _appID              : "",
  2949.   _appVersion         : "",
  2950.   _emDS               : null,
  2951.   _em                 : null,
  2952.   _versionUpdateOnly  : 0,
  2953.   _items              : [],
  2954.   
  2955.   /////////////////////////////////////////////////////////////////////////////
  2956.   // nsIExtensionItemUpdater
  2957.   //
  2958.   // When we check for updates to an item, there are two pieces of information
  2959.   // that are returned - 1) info about the newest available version, if any,
  2960.   // and 2) info about the currently installed version. The latter is provided
  2961.   // primarily to inform the client of changes to the application compatibility 
  2962.   // metadata for the current item. Depending on the situation, either 2 or 
  2963.   // 1&2 may be what is required.
  2964.   //
  2965.   // Callers:
  2966.   //  1 - nsUpdateService.js, user event
  2967.   //      User clicked on the update icon to invoke an update check, 
  2968.   //      user clicked on an Extension/Theme and clicked "Update". In this
  2969.   //      case we want to update compatibility metadata about the installed
  2970.   //      version, and look for newer versions to offer. 
  2971.   //  2 - nsUpdateService.js, background event
  2972.   //      Timer fired, background update is being performed. In this case
  2973.   //      we also want to update compatibility metadata and look for newer
  2974.   //      versions.
  2975.   //  3 - Mismatch
  2976.   //      User upgraded to a newer version of the app, update compatibility
  2977.   //      metadata and look for newer versions.
  2978.   //  4 - Install Phone Home
  2979.   //      User installed an item that was deemed incompatible based only
  2980.   //      on the information provided in the item's install.rdf manifest, 
  2981.   //      we look ONLY for compatibility updates in this case to determine
  2982.   //      whether or not the item can be installed.
  2983.   //  
  2984.   checkForUpdates: function nsExtensionItemUpdater_checkForUpdates (aItems, aItemCount, 
  2985.                                                                     aVersionUpdateOnly) 
  2986.   {
  2987.     gOS.notifyObservers(null, "Update:Extension:Started", "");
  2988.     this._versionUpdateOnly = aVersionUpdateOnly;
  2989.     this._items = aItems;
  2990.     this._responseCount = aItemCount;
  2991.     
  2992.     // This is the number of extensions/themes/etc that we found updates for.
  2993.     this._updateCount = 0;
  2994.  
  2995.     for (var i = 0; i < aItemCount; ++i) {
  2996.       var e = this._items[i];
  2997.       gOS.notifyObservers(e, "Update:Extension:Item-Started", "");
  2998.       (new nsRDFItemUpdater(this)).checkForUpdates(e, aVersionUpdateOnly);
  2999.     }
  3000.   },
  3001.   
  3002.   /////////////////////////////////////////////////////////////////////////////
  3003.   // nsExtensionItemUpdater
  3004.   _applyVersionUpdates: function nsExtensionItemUpdater__applyVersionUpdates (aLocalItem, aRemoteItem)
  3005.   {
  3006.     var r = this._emDS._getResourceForItem(aLocalItem.id);
  3007.     if (!r) return;
  3008.     var targetAppInfo = this._emDS.getTargetApplicationInfo(aLocalItem.id, this._emDS, 
  3009.                                                             getItemType(r.Value));
  3010.     if (gVersionChecker.compare(targetAppInfo.maxVersion, aRemoteItem.maxAppVersion) < 0) {
  3011.       // Remotely specified maxVersion is newer than the maxVersion 
  3012.       // for the installed Extension. Apply that change to the datasource.
  3013.       this._emDS.setTargetApplicationInfo(aLocalItem.id, 
  3014.                                           aRemoteItem.minAppVersion, 
  3015.                                           aRemoteItem.maxAppVersion,
  3016.                                           null, aLocalItem.type);
  3017.       
  3018.       // If we got here through |checkForMismatches|, this extension has
  3019.       // already been disabled, re-enable it.
  3020.       if (this._emDS.getItemProperty(aLocalItem.id, "disabled") == "true") 
  3021.         this._em.enableExtension(aLocalItem.id);
  3022.     }
  3023.   },
  3024.   
  3025.   _isValidUpdate: function nsExtensionItemUpdater__isValidUpdate (aLocalItem, aRemoteItem) 
  3026.   {
  3027.     var appExtensionsVersion = gPref.getCharPref(PREF_EM_APP_EXTENSIONS_VERSION);
  3028.  
  3029.     // Check if the update will only run on a newer version of Firefox. 
  3030.     if (aRemoteItem.minAppVersion && 
  3031.         gVersionChecker.compare(appExtensionsVersion, aRemoteItem.minAppVersion) < 0) 
  3032.       return false;
  3033.  
  3034.     // Check if the update will only run on an older version of Firefox. 
  3035.     if (aRemoteItem.maxAppVersion && 
  3036.         gVersionChecker.compare(appExtensionsVersion, aRemoteItem.maxAppVersion) > 0) 
  3037.       return false;
  3038.     
  3039.     return true;
  3040.   },
  3041.   
  3042.   _checkForDone: function nsExtensionItemUpdater__checkForDone ()
  3043.   {
  3044.     if (--this._responseCount == 0) {
  3045.       if (!this._versionUpdateOnly)
  3046.         gPref.setIntPref(PREF_UPDATE_COUNT, this._updateCount); 
  3047.           
  3048.       gOS.notifyObservers(null, "Update:Extension:Ended", "");
  3049.     }
  3050.   },
  3051.   
  3052.   /////////////////////////////////////////////////////////////////////////////
  3053.   // nsISupports
  3054.   QueryInterface: function nsExtensionItemUpdater_QueryInterface (aIID) 
  3055.   {
  3056.     if (!aIID.equals(Components.interfaces.nsIExtensionItemUpdater) &&
  3057.         !aIID.equals(Components.interfaces.nsISupports))
  3058.       throw Components.results.NS_ERROR_NO_INTERFACE;
  3059.     return this;
  3060.   }  
  3061. };
  3062.  
  3063. function nsRDFItemUpdater(aUpdater)
  3064. {
  3065.   this._updater = aUpdater;
  3066. }
  3067.  
  3068. nsRDFItemUpdater.prototype = {
  3069.   _updater            : null,
  3070.   _versionUpdateOnly  : 0,
  3071.   _item               : null,
  3072.   
  3073.   checkForUpdates: function (aItem, aVersionUpdateOnly)
  3074.   {
  3075.     // A preference setting can disable updating for this item
  3076.     try {
  3077.       if (!gPref.getBoolPref(PREF_EM_ITEM_UPDATE_ENABLED.replace(/%UUID%/, aItem.id))) {
  3078.         gOS.notifyObservers(null, "Update:Extension:Item-Ended", "");
  3079.         this._updater._checkForDone();
  3080.         return;
  3081.       }
  3082.     }
  3083.     catch (e) { }
  3084.  
  3085.     this._versionUpdateOnly = aVersionUpdateOnly;
  3086.     this._item = aItem;
  3087.   
  3088.     // Look for a custom update URI: 1) supplied by a pref, 2) supplied by the
  3089.     // install manifest, 3) the default configuration
  3090.     try {
  3091.       var dsURI = gPref.getComplexValue(PREF_EM_ITEM_UPDATE_URL.replace(/%UUID%/, aItem.id),
  3092.                                         Components.interfaces.nsIPrefLocalizedString).data;
  3093.     }
  3094.     catch (e) { }
  3095.     if (!dsURI)
  3096.       dsURI = aItem.updateRDF;
  3097.     if (!dsURI) {
  3098.       dsURI = gPref.getComplexValue(PREF_UPDATE_DEFAULT_URL,
  3099.                                     Components.interfaces.nsIPrefLocalizedString).data;
  3100.     }
  3101.     dsURI = dsURI.replace(/%ITEM_ID%/g, aItem.id);
  3102.     dsURI = dsURI.replace(/%ITEM_VERSION%/g, aItem.version);
  3103.     dsURI = dsURI.replace(/%ITEM_MAXAPPVERSION%/g, aItem.maxAppVersion);
  3104.     dsURI = dsURI.replace(/%APP_ID%/g, this._updater._appID);
  3105.     dsURI = dsURI.replace(/%APP_VERSION%/g, this._updater._appVersion);
  3106.     dsURI = dsURI.replace(/%REQ_VERSION%/g, 1);
  3107.     
  3108.     // escape() does not properly encode + symbols in any embedded FVF strings.
  3109.     dsURI = dsURI.replace(/\+/g, "%2B");
  3110.  
  3111.     var ds = gRDF.GetDataSource(dsURI);
  3112.     var rds = ds.QueryInterface(Components.interfaces.nsIRDFRemoteDataSource)
  3113.     if (rds.loaded)
  3114.       this.onDatasourceLoaded(ds, aItem);
  3115.     else {
  3116.       var sink = ds.QueryInterface(Components.interfaces.nsIRDFXMLSink);
  3117.       sink.addXMLSinkObserver(this);
  3118.     }
  3119.   },
  3120.  
  3121.   onDatasourceLoaded: function nsExtensionItemUpdater_onDatasourceLoaded (aDatasource, aLocalItem)
  3122.   {
  3123.     ///////////////////////////////////////////////////////////////////////////    
  3124.     // The extension update RDF file looks something like this:
  3125.     //
  3126.     //  <RDF:Description about="urn:mozilla:extension:{GUID}">
  3127.     //    <em:updates>
  3128.     //      <RDF:Seq>
  3129.     //        <RDF:li resource="urn:mozilla:extension:{GUID}:4.9"/>
  3130.     //        <RDF:li resource="urn:mozilla:extension:{GUID}:5.0"/>
  3131.     //      </RDF:Seq>
  3132.     //    </em:updates>
  3133.     //    <!-- the version of the extension being offered -->
  3134.     //    <em:version>5.0</em:version>
  3135.     //    <em:updateLink>http://www.mysite.com/myext-50.xpi</em:updateLink>
  3136.     //  </RDF:Description>
  3137.     //
  3138.     //  <RDF:Description about="urn:mozilla:extension:{GUID}:4.9">
  3139.     //    <em:version>4.9</em:version>
  3140.     //    <em:targetApplication>
  3141.     //      <RDF:Description>
  3142.     //        <em:id>{ec8030f7-c20a-464f-9b0e-13a3a9e97384}</em:id>
  3143.     //        <em:minVersion>0.9</em:minVersion>
  3144.     //        <em:maxVersion>1.0</em:maxVersion>
  3145.     //        <em:updateLink>http://www.mysite.com/myext-49.xpi</em:updateLink>
  3146.     //      </RDF:Description>
  3147.     //    </em:targetApplication>
  3148.     //  </RDF:Description>  
  3149.     //
  3150.     // If we get here because the following happened:
  3151.     // 1) User was using Firefox 0.9 with ExtensionX 0.5 (minVersion 0.8, 
  3152.     //    maxVersion 0.9 for Firefox)
  3153.     // 2) User upgraded Firefox to 1.0
  3154.     // 3) |checkForMismatches| deems ExtensionX 0.5 incompatible with this
  3155.     //    new version of Firefox on the basis of its maxVersion
  3156.     // 4) ** We reach this point **
  3157.     //
  3158.     // If the version of ExtensionX (0.5) matches that provided by the 
  3159.     // server, then this is a cue that the author updated the rdf file
  3160.     // or central repository to say "0.5 is ALSO compatible with Firefox 1.0,
  3161.     // no changes are necessary." In this event, the local metadata for
  3162.     // installed ExtensionX (0.5) is freshened with the new maxVersion, 
  3163.     // and we advance to the next item WITHOUT any download/install 
  3164.     // updates.
  3165.   
  3166.     // Parse the response RDF
  3167.     function UpdateData() {}; 
  3168.     UpdateData.prototype = { version: "0.0", updateLink: null, 
  3169.                              minVersion: "0.0", maxVersion: "0.0" };
  3170.     
  3171.     var versionUpdate = new UpdateData();
  3172.     var newestUpdate  = new UpdateData();
  3173.  
  3174.     var newerItem, sameItem;
  3175.     
  3176.     // Firefox 1.0PR+ update.rdf format
  3177.     if (!this._versionUpdateOnly) {
  3178.       // Look for newer versions of this item, we only do this in "normal" 
  3179.       // mode... see comment by nsExtensionItemUpdater_checkForUpdates 
  3180.       // about how we do this in all cases but Install Phone Home - which 
  3181.       // only needs to do a version check.
  3182.       this._parseV20UpdateInfo(aDatasource, aLocalItem, newestUpdate, false);
  3183.       if (!newestUpdate.updateLink) {
  3184.         // Firefox 0.9 update.rdf format - does not contain any metadata
  3185.         // that can be used for version updates, so performed in the "all updates"
  3186.         // mode only. 
  3187.         this._parseV10UpdateInfo(aDatasource, aLocalItem, newestUpdate);
  3188.       }
  3189.  
  3190.       newerItem = Components.classes["@mozilla.org/updates/item;1"]
  3191.                             .createInstance(Components.interfaces.nsIUpdateItem);
  3192.       newerItem.init(aLocalItem.id, 
  3193.                      newestUpdate.version, 
  3194.                      newestUpdate.minVersion, 
  3195.                      newestUpdate.maxVersion, 
  3196.                      aLocalItem.name, 
  3197.                      -1, newestUpdate.updateLink, "", "", 
  3198.                      aLocalItem.type);
  3199.       if (this._updater._isValidUpdate(aLocalItem, newerItem))
  3200.         ++this._updater._updateCount;
  3201.       else
  3202.         newerItem = null;
  3203.     }
  3204.     
  3205.     // Now look for updated version compatibility metadata for the currently
  3206.     // installed version...
  3207.     this._parseV20UpdateInfo(aDatasource, aLocalItem, versionUpdate, true);
  3208.  
  3209.     var result = gVersionChecker.compare(versionUpdate.version, 
  3210.                                           aLocalItem.version);
  3211.     if (result == 0) {
  3212.       // Local version exactly matches the "Version Update" remote version, 
  3213.       // Apply changes into local datasource.
  3214.       sameItem = Components.classes["@mozilla.org/updates/item;1"]
  3215.                            .createInstance(Components.interfaces.nsIUpdateItem);
  3216.       sameItem.init(aLocalItem.id, 
  3217.                     versionUpdate.version, 
  3218.                     versionUpdate.minVersion, 
  3219.                     versionUpdate.maxVersion, 
  3220.                     aLocalItem.name, 
  3221.                     -1, "", "", "", 
  3222.                     aLocalItem.type);
  3223.       if (!this._versionUpdateOnly) {
  3224.         if (this._updater._isValidUpdate(aLocalItem, sameItem)) {
  3225.           // Install-time updates are not written to the DS because there is no
  3226.           // entry yet, EM just uses the notifications to ascertain (by hand)
  3227.           // whether or not there is a remote maxVersion tweak that makes the 
  3228.           // item being installed compatible.
  3229.           this._updater._applyVersionUpdates(aLocalItem, sameItem);
  3230.         }
  3231.         else 
  3232.           sameItem = null;
  3233.       }
  3234.     }
  3235.     
  3236.     gOS.notifyObservers(!this._versionUpdateOnly ? newerItem : sameItem, 
  3237.                         "Update:Extension:Item-Ended", "");
  3238.     
  3239.     // Only one call of this._updater._checkForDone is needed for RDF 
  3240.     // responses, since there is only one response per item.
  3241.     this._updater._checkForDone();
  3242.   },
  3243.  
  3244.   // Parses Firefox 0.9 update.rdf format  
  3245.   _parseV10UpdateInfo: function nsExtensionItemUpdater__parseV10UpdateInfo (aDataSource, aLocalItem, aUpdateData)
  3246.   {
  3247.     var extensionRes  = gRDF.GetResource(getItemPrefix(aLocalItem.type) + aLocalItem.id);
  3248.     
  3249.     aUpdateData.version     = this._getPropertyFromResource(aDataSource, extensionRes, 
  3250.                                                             "version", aLocalItem);
  3251.     aUpdateData.updateLink  = this._getPropertyFromResource(aDataSource, extensionRes, 
  3252.                                                             "updateLink", aLocalItem);
  3253.   },
  3254.   
  3255.   // Get a compulsory property from a resource. Reports an error if the 
  3256.   // property was not present. 
  3257.   _getPropertyFromResource: function nsExtensionItemUpdater__getPropertyFromResource (aDataSource,
  3258.                                                                                       aSourceResource, 
  3259.                                                                                       aProperty, 
  3260.                                                                                       aLocalItem)
  3261.   {
  3262.     var rv;
  3263.     try {
  3264.       var property = gRDF.GetResource(EM_NS(aProperty));
  3265.       rv = stringData(aDataSource.GetTarget(aSourceResource, property, true));
  3266.       if (rv == "--")
  3267.         throw Components.results.NS_ERROR_FAILURE;
  3268.     }
  3269.     catch (e) { 
  3270.       // XXXben show console message "aProperty" not found on aSourceResource. 
  3271.       return null;
  3272.     }
  3273.     return rv;
  3274.   },
  3275.   
  3276.   // Parses Firefox 1.0RC1+ update.rdf format
  3277.   _parseV20UpdateInfo: function nsExtensionItemUpdater__parseV20UpdateInfo (aDataSource, 
  3278.                                                                             aLocalItem, 
  3279.                                                                             aUpdateData, 
  3280.                                                                             aVersionUpdatesOnly)
  3281.   {
  3282.     var extensionRes  = gRDF.GetResource(getItemPrefix(aLocalItem.type) + aLocalItem.id);
  3283.  
  3284.     var updatesArc = gRDF.GetResource(EM_NS("updates"));
  3285.     var updates = aDataSource.GetTarget(extensionRes, updatesArc, true);
  3286.     
  3287.     try {
  3288.       updates = updates.QueryInterface(Components.interfaces.nsIRDFResource);
  3289.     }
  3290.     catch (e) { return; }
  3291.     
  3292.     var cu = Components.classes["@mozilla.org/rdf/container-utils;1"]
  3293.                        .getService(Components.interfaces.nsIRDFContainerUtils);
  3294.     if (cu.IsContainer(aDataSource, updates)) {
  3295.       var c = Components.classes["@mozilla.org/rdf/container;1"]
  3296.                         .getService(Components.interfaces.nsIRDFContainer);
  3297.       c.Init(aDataSource, updates);
  3298.  
  3299.       // In "all update types" mode, we look for newer versions, starting with the 
  3300.       // current installed version.
  3301.       if (!aVersionUpdatesOnly) 
  3302.         aUpdateData.version = aLocalItem.version;
  3303.  
  3304.       var versions = c.GetElements();
  3305.       while (versions.hasMoreElements()) {
  3306.         // There are two different methodologies for collecting version 
  3307.         // information depending on whether or not we've bene invoked in 
  3308.         // "version updates only" mode or "version+newest" mode. 
  3309.         var version = versions.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3310.         this._parseV20Update(aDataSource, version, aLocalItem, aUpdateData, aVersionUpdatesOnly);
  3311.         if (aVersionUpdatesOnly && aUpdateData.updateLink)
  3312.           break;
  3313.       }
  3314.     }
  3315.   },
  3316.   
  3317.   _parseV20Update: function nsExtensionItemUpdater__parseV20Update (aDataSource,
  3318.                                                                     aUpdateResource,
  3319.                                                                     aLocalItem,
  3320.                                                                     aUpdateData,
  3321.                                                                     aVersionUpdatesOnly)
  3322.   {
  3323.     var version = this._getPropertyFromResource(aDataSource, aUpdateResource, 
  3324.                                                 "version", aLocalItem);
  3325.     var taArc = gRDF.GetResource(EM_NS("targetApplication"));
  3326.     var targetApps = aDataSource.GetTargets(aUpdateResource, taArc, true);
  3327.     while (targetApps.hasMoreElements()) {
  3328.       var targetApp = targetApps.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3329.       var id = this._getPropertyFromResource(aDataSource, targetApp, "id", aLocalItem);
  3330.       if (id != this._updater._appID)
  3331.         continue;
  3332.       
  3333.       var result = gVersionChecker.compare(version, aLocalItem.version);
  3334.       if (aVersionUpdatesOnly ? result == 0 : result > 0) {
  3335.         aUpdateData.version = version;
  3336.         aUpdateData.updateLink = this._getPropertyFromResource(aDataSource, targetApp, "updateLink", aLocalItem);
  3337.         aUpdateData.minVersion = this._getPropertyFromResource(aDataSource, targetApp, "minVersion", aLocalItem);
  3338.         aUpdateData.maxVersion = this._getPropertyFromResource(aDataSource, targetApp, "maxVersion", aLocalItem);
  3339.       }
  3340.     }
  3341.   },
  3342.   
  3343.   onDatasourceError: function nsExtensionItemUpdater_onDatasourceError (aItem, aError)
  3344.   {
  3345.     gOS.notifyObservers(aItem, "Update:Extension:Item-Error", aError);
  3346.     this._updater._checkForDone();
  3347.   },
  3348.  
  3349.   /////////////////////////////////////////////////////////////////////////////
  3350.   // nsIRDFXMLSinkObserver
  3351.   onBeginLoad: function(aSink)
  3352.   {
  3353.   },
  3354.   onInterrupt: function(aSink)
  3355.   {
  3356.   },
  3357.   onResume: function(aSink)
  3358.   {
  3359.   },
  3360.   
  3361.   onEndLoad: function(aSink)
  3362.   {
  3363.     try {
  3364.       aSink.removeXMLSinkObserver(this);
  3365.       
  3366.       var ds = aSink.QueryInterface(Components.interfaces.nsIRDFDataSource);
  3367.       this.onDatasourceLoaded(ds, this._item);
  3368.     }
  3369.     catch (e) { }
  3370.   },
  3371.   
  3372.   onError: function(aSink, aStatus, aErrorMsg)
  3373.   {
  3374.     try {
  3375.       aSink.removeXMLSinkObserver(this);
  3376.       
  3377.       this.onDatasourceError(this._item, aStatus.toString());
  3378.     }
  3379.     catch (e) { }
  3380.   }
  3381. };
  3382.  
  3383. ///////////////////////////////////////////////////////////////////////////////
  3384. //
  3385. // nsExtensionsDataSource
  3386. //
  3387. function nsExtensionsDataSource()
  3388. {
  3389. }
  3390.  
  3391. nsExtensionsDataSource.prototype = {
  3392.   _appExtensions      : null,
  3393.   _profileExtensions  : null,  
  3394.   _composite          : null,
  3395.   safeMode            : false,
  3396.   
  3397.   _emR: function nsExtensionsDataSource__emR (aProperty) 
  3398.   {
  3399.     return gRDF.GetResource(EM_NS(aProperty));
  3400.   },
  3401.   
  3402.   _emL: function nsExtensionsDataSource__emL (aLiteral)
  3403.   {
  3404.     return gRDF.GetLiteral(aLiteral);
  3405.   },
  3406.   
  3407.   isCompatible: function nsExtensionsDataSource__isCompatible (aDS, aSource)
  3408.   {
  3409.     // XXXben - cheap hack. Our bundled items are always compatible. 
  3410.     if (aSource.EqualsNode(gRDF.GetResource(getItemPrefix(nsIUpdateItem.TYPE_THEME) + "{972ce4c6-7e08-4474-a285-3208198ce6fd}")) || 
  3411.         aSource.EqualsNode(gRDF.GetResource(getItemPrefix(nsIUpdateItem.TYPE_EXTENSION) + "{641d8d09-7dda-4850-8228-ac0ab65e2ac9}")))
  3412.       return true;
  3413.       
  3414.     var appVersion = gPref.getCharPref(PREF_EM_APP_EXTENSIONS_VERSION);
  3415.     var appID = gPref.getCharPref(PREF_EM_APP_ID);
  3416.     
  3417.     var targets = aDS.GetTargets(aSource, this._emR("targetApplication"), true);
  3418.     var idRes = this._emR("id");
  3419.     var minVersionRes = this._emR("minVersion");
  3420.     var maxVersionRes = this._emR("maxVersion");
  3421.     while (targets.hasMoreElements()) {
  3422.       var targetApp = targets.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3423.       var id          = stringData(aDS.GetTarget(targetApp, idRes, true));
  3424.       var minVersion  = stringData(aDS.GetTarget(targetApp, minVersionRes, true));
  3425.       var maxVersion  = stringData(aDS.GetTarget(targetApp, maxVersionRes, true));
  3426.  
  3427.       if (id == appID) {
  3428.         var versionChecker = getVersionChecker();
  3429.         return ((versionChecker.compare(appVersion, minVersion) >= 0) &&
  3430.                 (versionChecker.compare(appVersion, maxVersion) <= 0));
  3431.       }
  3432.     }
  3433.     return false;
  3434.   },
  3435.   
  3436.   getIncompatibleItemList: function nsExtensionsDataSource_getIncompatibleItemList (aAppID, aAppVersion, aItemType)
  3437.   {
  3438.     var items = [];
  3439.     var roots = getItemRoots(aItemType);
  3440.     for (var i = 0; i < roots.length; ++i) {    
  3441.       var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  3442.                           .createInstance(Components.interfaces.nsIRDFContainer);
  3443.       ctr.Init(this._composite, gRDF.GetResource(roots[i]));
  3444.       
  3445.       var elements = ctr.GetElements();
  3446.       while (elements.hasMoreElements()) {
  3447.         var e = elements.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3448.         var itemType = getItemType(e.Value);
  3449.         if (itemType != -1 && !this.isCompatible(this, e))
  3450.           items.push(this.getItemForID(stripPrefix(e.Value, itemType)));
  3451.       }
  3452.     }
  3453.     return items;
  3454.   },
  3455.   
  3456.   getItemList: function nsExtensionsDataSource_getItemList(aItemID, aType, aCountRef)
  3457.   {
  3458.     var items = [];
  3459.     if (aItemID)
  3460.       items.push(this.getItemForID(aItemID));
  3461.     else {
  3462.       var roots = getItemRoots(aType);
  3463.       for (var i = 0; i < roots.length; ++i) {
  3464.         var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  3465.                             .createInstance(Components.interfaces.nsIRDFContainer);
  3466.         ctr.Init(this, gRDF.GetResource(roots[i]));
  3467.         
  3468.         var elements = ctr.GetElements();
  3469.         while (elements.hasMoreElements()) {
  3470.           var e = elements.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3471.           var itemType = getItemType(e.Value);
  3472.           if (itemType != -1)
  3473.             items.push(this.getItemForID(stripPrefix(e.Value, itemType)));
  3474.         }
  3475.       }
  3476.     }
  3477.     aCountRef.value = items.length;
  3478.     return items;
  3479.   },
  3480.  
  3481.   // XXXben this function is a little weird since it returns an array of strings, not
  3482.   // an array of nsIUpdateItems...  
  3483.   getItemsWithFlagSet: function nsExtensionsDataSource_getItemsWithFlagSet (aFlag)
  3484.   {
  3485.     var items = [];
  3486.     var sources = this.GetSources(this._emR(aFlag), this._emL("true"), true);
  3487.     while (sources.hasMoreElements()) {
  3488.       var e = sources.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3489.       
  3490.       items.push(stripPrefix(e.Value, getItemType(e.Value)));
  3491.     }
  3492.     return items;
  3493.   },
  3494.   
  3495.   getItemsWithFlagUnset: function nsExtensionsDataSource_getItemsWithFlagUnset (aFlag, aItemType)
  3496.   {
  3497.     var items = [];
  3498.     
  3499.     var roots = getItemRoots(aItemType);
  3500.     for (var i = 0; i < roots.length; ++i) {
  3501.       var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  3502.                           .createInstance(Components.interfaces.nsIRDFContainer);
  3503.       ctr.Init(this, gRDF.GetResource(roots[i]));
  3504.       
  3505.       var elements = ctr.GetElements();
  3506.       while (elements.hasMoreElements()) {
  3507.         var e = elements.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3508.         if (getItemType(e.Value) != -1) {
  3509.           var value = this.GetTarget(e, this._emR(aFlag), true);
  3510.           if (!value)
  3511.             items.push(stripPrefix(e.Value, getItemType(e.Value)));
  3512.         }
  3513.       }
  3514.     }
  3515.     return items;
  3516.   },
  3517.   
  3518.   getItemForID: function nsExtensionsDataSource_getItemForID (aItemID)
  3519.   {
  3520.     var item = Components.classes["@mozilla.org/updates/item;1"]
  3521.                          .createInstance(Components.interfaces.nsIUpdateItem);
  3522.                          
  3523.     var r = this._getResourceForItem(aItemID);
  3524.     if (!r)
  3525.       return null;
  3526.     
  3527.     var targetAppInfo = this.getTargetApplicationInfo(aItemID, this, getItemType(r.Value));
  3528.     item.init(aItemID, 
  3529.               this.getItemProperty(aItemID, "version"), 
  3530.               targetAppInfo ? targetAppInfo.minVersion : "",
  3531.               targetAppInfo ? targetAppInfo.maxVersion : "",
  3532.               this.getItemProperty(aItemID, "name"),
  3533.               -1, 
  3534.               "", /* XPI Update URL */
  3535.               this.getItemProperty(aItemID, "iconURL"), 
  3536.               this.getItemProperty(aItemID, "updateURL"), 
  3537.               getItemType(r.Value));
  3538.     return item;
  3539.   },
  3540.   
  3541.   isProfileItem: function nsExtensionsDataSource_isProfileItem (aItemID)
  3542.   {
  3543.     return this.getItemProperty(aItemID, "installLocation") != "global";
  3544.   },
  3545.   
  3546.   _setProperty: function nsExtensionsDataSource__setProperty (aDS, aSource, aProperty, aNewValue)
  3547.   {
  3548.     var oldValue = aDS.GetTarget(aSource, aProperty, true);
  3549.     if (oldValue) {
  3550.       if (aNewValue)
  3551.         aDS.Change(aSource, aProperty, oldValue, aNewValue);
  3552.       else
  3553.         aDS.Unassert(aSource, aProperty, oldValue);
  3554.     }
  3555.     else if (aNewValue)
  3556.       aDS.Assert(aSource, aProperty, aNewValue, true);
  3557.   },
  3558.   
  3559.   // Given a GUID, get the RDF resource representing the item. This
  3560.   // will be of the form urn:mozilla:extension:{GUID} or 
  3561.   // urn:mozilla:theme:{GUID} depending on the item type. 
  3562.   _getResourceForItem: function nsExtensionsDataSource__getResourceForItem(aItemID)
  3563.   {
  3564.     var res = null;
  3565.     
  3566.     // We can try and infer the resource URI from presence in one of the 
  3567.     // item lists.
  3568.     var rdfc = Components.classes["@mozilla.org/rdf/container;1"]
  3569.                          .createInstance(Components.interfaces.nsIRDFContainer);
  3570.     rdfc.Init(this, gRDF.GetResource(ROOT_EXTENSION));
  3571.     res = gRDF.GetResource(PREFIX_EXTENSION + aItemID);
  3572.     if (rdfc.IndexOf(res) != -1)
  3573.       return res;
  3574.     
  3575.     rdfc.Init(this, gRDF.GetResource(ROOT_THEME));
  3576.     res = gRDF.GetResource(PREFIX_THEME + aItemID);
  3577.     if (rdfc.IndexOf(res) != -1)
  3578.       return res;
  3579.  
  3580.     return null;
  3581.   },
  3582.  
  3583.   getTargetApplicationInfo: function nsExtensionsDataSource_getTargetApplicationInfo (aExtensionID, aDataSource, aType)
  3584.   {
  3585.     var internalName = this.getItemProperty(aExtensionID, "internalName");
  3586.     // The default theme is always compatible. 
  3587.     if (internalName == KEY_DEFAULT_THEME) {
  3588.       var ver = gPref.getCharPref(PREF_EM_APP_EXTENSIONS_VERSION);
  3589.       return { minVersion: ver, maxVersion: ver };
  3590.     }
  3591.  
  3592.     var appID = gPref.getCharPref(PREF_EM_APP_ID);
  3593.     var r = gRDF.GetResource(getItemPrefix(aType) + aExtensionID);
  3594.     var targetApps = aDataSource.GetTargets(r, this._emR("targetApplication"), true);
  3595.     if (!targetApps.hasMoreElements()) {
  3596.       r = gRDF.GetResource("urn:mozilla:install-manifest"); 
  3597.       targetApps = aDataSource.GetTargets(r, this._emR("targetApplication"), true); 
  3598.     }
  3599.     while (targetApps.hasMoreElements()) {
  3600.       var targetApp = targetApps.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3601.       if (targetApp) {
  3602.         try {
  3603.           var id = stringData(aDataSource.GetTarget(targetApp, this._emR("id"), true));
  3604.           if (id != appID) // Different target application
  3605.             continue;
  3606.           
  3607.           return { minVersion: stringData(aDataSource.GetTarget(targetApp, this._emR("minVersion"), true)),
  3608.                    maxVersion: stringData(aDataSource.GetTarget(targetApp, this._emR("maxVersion"), true)) };
  3609.         }
  3610.         catch (e) { 
  3611.           continue;
  3612.         }
  3613.       }
  3614.     }
  3615.     return null;
  3616.   },
  3617.   
  3618.   setTargetApplicationInfo: function nsExtensionsDataSource_setTargetApplicationInfo(aExtensionID, aMinVersion, 
  3619.                                                                                      aMaxVersion, aDataSource, aType)
  3620.   {
  3621.     var targetDataSource = aDataSource;
  3622.     if (!targetDataSource)
  3623.       targetDataSource = this;
  3624.       
  3625.     var appID = gPref.getCharPref(PREF_EM_APP_ID);
  3626.     var r = gRDF.GetResource(getItemPrefix(aType) + aExtensionID);
  3627.     var targetApps = targetDataSource.GetTargets(r, this._emR("targetApplication"), true);
  3628.     if (!targetApps.hasMoreElements()) {
  3629.       r = gRDF.GetResource("urn:mozilla:install-manifest"); 
  3630.       targetApps = aDataSource.GetTargets(r, this._emR("targetApplication"), true); 
  3631.     }
  3632.     while (targetApps.hasMoreElements()) {
  3633.       var targetApp = targetApps.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3634.       if (targetApp) {
  3635.         var id = stringData(targetDataSource.GetTarget(targetApp, this._emR("id"), true));
  3636.         if (id != appID) // Different target application
  3637.           continue;
  3638.         
  3639.         if (!aDataSource) {
  3640.           var isProfile = this.isProfileItem(aExtensionID);
  3641.           targetDataSource = isProfile ? this._profileExtensions : this._appExtensions;
  3642.         }
  3643.         this._setProperty(targetDataSource, targetApp, this._emR("minVersion"), this._emL(aMinVersion));
  3644.         this._setProperty(targetDataSource, targetApp, this._emR("maxVersion"), this._emL(aMaxVersion));
  3645.         
  3646.         if (!aDataSource)
  3647.           this._flush(isProfile);
  3648.       }
  3649.     }
  3650.   },
  3651.   
  3652.   getItemProperty: function nsExtensionsDataSource_getItemProperty (aItemID, aProperty)
  3653.   { 
  3654.     var item = this._getResourceForItem(aItemID);
  3655.     if (!item) {
  3656.       dump("*** getItemProperty failing for lack of an item. This means getResourceForItem \
  3657.                 failed to locate a resource for aItemID (item ID = " + aItemID + ", property = " + aProperty + ")\n");
  3658.     }
  3659.     else 
  3660.       return this._getItemProperty(item, aProperty);
  3661.     return undefined;
  3662.   },
  3663.   
  3664.   _getItemProperty: function nsExtensionsDataSource__getItemProperty (aItemResource, aProperty)
  3665.   {
  3666.     try {
  3667.       return this.GetTarget(aItemResource, this._emR(aProperty), true).QueryInterface(Components.interfaces.nsIRDFLiteral).Value;
  3668.     }
  3669.     catch (e) {}
  3670.     return "";
  3671.   },
  3672.   
  3673.   setItemProperty: function nsExtensionsDataSource_setItemProperty(
  3674.     aItemID, aPropertyArc, aPropertyValue, aIsProfile, aItemType)
  3675.   {
  3676.     var item = gRDF.GetResource(getItemPrefix(aItemType) + aItemID);
  3677.     var ds = aIsProfile ? this._profileExtensions : this._appExtensions;
  3678.     this._setProperty(ds, item, aPropertyArc, aPropertyValue);
  3679.  
  3680.     this._flush(aIsProfile);  
  3681.   },
  3682.  
  3683.   insertForthcomingItem: function nsExtensionsDataSource_insertForthcomingItem (aItemID, aItemType, aIsProfile)
  3684.   {
  3685.     // Get the target container and resource
  3686.     var targetDS = aIsProfile ? this._profileExtensions : this._appExtensions;
  3687.     var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  3688.                         .createInstance(Components.interfaces.nsIRDFContainer);
  3689.     ctr.Init(targetDS, gRDF.GetResource(getItemRoot(aItemType)));
  3690.  
  3691.     var targetRes = gRDF.GetResource(getItemPrefix(aItemType) + aItemID);
  3692.     // Don't bother adding the extension to the list if it's already there. 
  3693.     // (i.e. we're upgrading)
  3694.     var oldIndex = ctr.IndexOf(targetRes);
  3695.     if (oldIndex == -1)
  3696.       ctr.AppendElement(targetRes);
  3697.  
  3698.     this._flush(aIsProfile);
  3699.   }, 
  3700.  
  3701.   removeItemFromContainer: function nsExtensionsDataSource_removeItemFromContainer(aItemID, aItemType, aIsProfile)
  3702.   {
  3703.     var targetDS = aIsProfile ? this._profileExtensions : this._appExtensions;
  3704.     var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  3705.                         .createInstance(Components.interfaces.nsIRDFContainer);
  3706.     ctr.Init(targetDS, gRDF.GetResource(getItemRoot(aItemType)));
  3707.     
  3708.     var item = gRDF.GetResource(getItemPrefix(aItemType) + aItemID);
  3709.     ctr.RemoveElement(item, true);
  3710.     
  3711.     this._flush(aIsProfile);
  3712.   },
  3713.  
  3714.   // Removes a corrupt item entry from the extension list added due to 
  3715.   // buggy code in previous EM versions!  
  3716.   removeCorruptItem: function nsExtensionsDataSource_removeCorruptItem (aItemID, aItemType, aIsProfile)
  3717.   {
  3718.     this.removeItemMetadata(aItemID, aItemType);
  3719.     this.removeItemFromContainer(aItemID, aItemType, aIsProfile);
  3720.   },
  3721.  
  3722.   // Removes a corrupt download entry from the list.   
  3723.   removeCorruptDLItem: function nsExtensionsDataSource_removeCorruptDLItem (aItemURI, aItemType)
  3724.   {
  3725.     var itemResource = gRDF.GetResource(aItemURI);
  3726.     var itemRoot = gRDF.GetResource(getItemRoot(aItemType));
  3727.     var dses = [this._profileExtensions, this._appExtensions];
  3728.     var isProfile = [true, false];
  3729.     for (var i = 0; i < dses.length; ++i) {
  3730.       var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  3731.                           .createInstance(Components.interfaces.nsIRDFContainer);
  3732.       ctr.Init(dses[i], itemRoot);
  3733.       if (ctr.IndexOf(itemResource) != -1) {
  3734.         ctr.RemoveElement(itemResource, true);
  3735.         this._cleanResource(itemResource, dses[i]);
  3736.         this._flush(isProfile[i]);
  3737.         break;
  3738.       }
  3739.     }
  3740.   },
  3741.   
  3742.   addItemMetadata: function nsExtensionsDataSource_addItemMetadata (aItemID, aItemType, aSourceDS, aIsProfile)
  3743.   {
  3744.     var targetDS = aIsProfile ? this._profileExtensions : this._appExtensions;
  3745.     var targetRes = gRDF.GetResource(getItemPrefix(aItemType) + aItemID);
  3746.  
  3747.     // Copy the assertions over from the source datasource. 
  3748.     
  3749.     // Assert properties with single values
  3750.     var singleProps = ["version", "name", "description", "creator", "homepageURL", 
  3751.                        "updateURL", "updateService", "optionsURL", "aboutURL", 
  3752.                        "iconURL", "internalName"];
  3753.  
  3754.     // Global extensions and themes can also be locked (can't be removed or disabled).
  3755.     if (!aIsProfile)
  3756.       singleProps = singleProps.concat(["locked"]);
  3757.     var sourceRes = gRDF.GetResource("urn:mozilla:install-manifest");
  3758.     for (var i = 0; i < singleProps.length; ++i) {
  3759.       var property = this._emR(singleProps[i]);
  3760.       var literal = aSourceDS.GetTarget(sourceRes, property, true);
  3761.       if (!literal)
  3762.         continue; // extension didn't specify this property, no big deal, continue.
  3763.         
  3764.       var val = literal.QueryInterface(Components.interfaces.nsIRDFLiteral).Value;
  3765.       
  3766.       var oldValue = targetDS.GetTarget(targetRes, property, true);
  3767.       if (!oldValue)
  3768.         targetDS.Assert(targetRes, property, literal, true);
  3769.       else
  3770.         targetDS.Change(targetRes, property, oldValue, literal);
  3771.     }    
  3772.  
  3773.     // Assert properties with multiple values    
  3774.     var manyProps = ["contributor"];
  3775.     for (var i = 0; i < singleProps.length; ++i) {
  3776.       var property = this._emR(manyProps[i]);
  3777.       var literals = aSourceDS.GetTargets(sourceRes, property, true);
  3778.       
  3779.       var oldValues = targetDS.GetTargets(targetRes, property, true);
  3780.       while (oldValues.hasMoreElements()) {
  3781.         var oldValue = oldValues.getNext().QueryInterface(Components.interfaces.nsIRDFNode);
  3782.         targetDS.Unassert(targetRes, property, oldValue);
  3783.       }
  3784.       while (literals.hasMoreElements()) {
  3785.         var literal = literals.getNext().QueryInterface(Components.interfaces.nsIRDFNode);
  3786.         targetDS.Assert(targetRes, property, literal, true);
  3787.       }
  3788.     }
  3789.     
  3790.     // Version/Dependency Info
  3791.     var versionProps = ["targetApplication", "requires"];
  3792.     var idRes = this._emR("id");
  3793.     var minVersionRes = this._emR("minVersion");
  3794.     var maxVersionRes = this._emR("maxVersion");
  3795.     for (var i = 0; i < versionProps.length; ++i) {
  3796.       var property = this._emR(versionProps[i]);
  3797.       var newVersionInfos = aSourceDS.GetTargets(sourceRes, property, true);
  3798.       
  3799.       var oldVersionInfos = targetDS.GetTargets(targetRes, property, true);
  3800.       while (oldVersionInfos.hasMoreElements()) {
  3801.         var oldVersionInfo = oldVersionInfos.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3802.         this._cleanResource(oldVersionInfo, targetDS);
  3803.         targetDS.Unassert(targetRes, property, oldVersionInfo);
  3804.       }
  3805.       while (newVersionInfos.hasMoreElements()) {
  3806.         var newVersionInfo = newVersionInfos.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3807.         var anon = gRDF.GetAnonymousResource();
  3808.         targetDS.Assert(anon, idRes, aSourceDS.GetTarget(newVersionInfo, idRes, true), true);
  3809.         targetDS.Assert(anon, minVersionRes, aSourceDS.GetTarget(newVersionInfo, minVersionRes, true), true);
  3810.         targetDS.Assert(anon, maxVersionRes, aSourceDS.GetTarget(newVersionInfo, maxVersionRes, true), true);
  3811.         targetDS.Assert(targetRes, property, anon, true);
  3812.       }
  3813.     }
  3814.     
  3815.     this._flush(aIsProfile);
  3816.   },
  3817.   
  3818.   lockUnlockItem: function nsExtensionsDataSource_lockUnlockItem (aItemID, aLocked)
  3819.   {
  3820.     var item = this._getResourceForItem(aItemID);
  3821.     if (item) {
  3822.       var val = aLocked ? this._emL("true") : this._emL("false");
  3823.       this.setItemProperty(aItemID, this._emR("locked"), val, false, getItemType(item.Value));
  3824.       this._flush(false);
  3825.     }
  3826.   },  
  3827.   
  3828.   enableExtension: function nsExtensionsDataSource_enableExtension (aExtensionID)
  3829.   {
  3830.     this.setItemProperty(aExtensionID, this._emR("toBeEnabled"), 
  3831.                          this._emL("true"), this.isProfileItem(aExtensionID), 
  3832.                          nsIUpdateItem.TYPE_EXTENSION);
  3833.     this.setItemProperty(aExtensionID, this._emR("toBeDisabled"), 
  3834.                          null, this.isProfileItem(aExtensionID), 
  3835.                          nsIUpdateItem.TYPE_EXTENSION);
  3836.     this.setItemProperty(aExtensionID, this._emR("disabled"), 
  3837.                          null, this.isProfileItem(aExtensionID), 
  3838.                          nsIUpdateItem.TYPE_EXTENSION);
  3839.   },
  3840.   
  3841.   disableExtension: function nsExtensionsDataSource_disableExtension (aExtensionID)
  3842.   {
  3843.     this.setItemProperty(aExtensionID, this._emR("toBeDisabled"), 
  3844.                          this._emL("true"), this.isProfileItem(aExtensionID), 
  3845.                          nsIUpdateItem.TYPE_EXTENSION);
  3846.     this.setItemProperty(aExtensionID, this._emR("toBeEnabled"), 
  3847.                          null, this.isProfileItem(aExtensionID), 
  3848.                          nsIUpdateItem.TYPE_EXTENSION);
  3849.     this.setItemProperty(aExtensionID, this._emR("disabled"), 
  3850.                          this._emL("true"), this.isProfileItem(aExtensionID), 
  3851.                          nsIUpdateItem.TYPE_EXTENSION);
  3852.   },
  3853.   
  3854.   uninstallExtension: function nsExtensionsDataSource_uninstallExtension (aExtensionID)
  3855.   {
  3856.     // We have to do this check BEFORE we unhook all the metadata from this 
  3857.     // extension's resource, otherwise we'll think it's a global extension.
  3858.     var isProfile = this.isProfileItem(aExtensionID);
  3859.     
  3860.     this.setItemProperty(aExtensionID, this._emR("toBeInstalled"), 
  3861.                          null, isProfile, 
  3862.                          nsIUpdateItem.TYPE_EXTENSION);
  3863.     this.setItemProperty(aExtensionID, this._emR("toBeUninstalled"), 
  3864.                          this._emL("true"), isProfile, 
  3865.                          nsIUpdateItem.TYPE_EXTENSION);
  3866.     this._flush(isProfile);
  3867.   },
  3868.   
  3869.   doneInstallingTheme: function nsExtensionsDataSource_doneInstallingTheme (aThemeID)
  3870.   {
  3871.     // Notify observers of a change in the iconURL property to cause the UI to
  3872.     // refresh.
  3873.     var theme = this._getResourceForItem(aThemeID);
  3874.     var iconURLArc = this._emR("iconURL");
  3875.     var iconURL = this.GetTarget(theme, iconURLArc, true);
  3876.     if (theme, iconURLArc, iconURL) {
  3877.       for (var i = 0; i < this._observers.length; ++i)
  3878.         this._observers[i].onAssert(this, theme, iconURLArc, iconURL);
  3879.     }
  3880.   },
  3881.   
  3882.   enableTheme: function nsExtensionsDataSource_enableTheme (aThemeID)
  3883.   {
  3884.     this.setItemProperty(aThemeID, this._emR("disabled"), 
  3885.                          null, this.isProfileItem(aThemeID), 
  3886.                          nsIUpdateItem.TYPE_THEME);
  3887.   },
  3888.     
  3889.   disableTheme: function nsExtensionsDataSource_disableTheme (aThemeID)
  3890.   {
  3891.     this.setItemProperty(aThemeID, this._emR("disabled"), 
  3892.                          this._emL("true"), this.isProfileItem(aThemeID), 
  3893.                          nsIUpdateItem.TYPE_THEME);
  3894.   },
  3895.   
  3896.   uninstallTheme: function nsExtensionsDataSource_uninstallTheme(aThemeID)
  3897.   {
  3898.     // We have to do this check BEFORE we unhook all the metadata from this 
  3899.     // extension's resource, otherwise we'll think it's a global extension.
  3900.     var isProfile = this.isProfileItem(aThemeID);
  3901.         
  3902.     // Clean the extension resource
  3903.     this.removeItemMetadata(aThemeID, nsIUpdateItem.TYPE_THEME);
  3904.     
  3905.     var uninstaller = new nsThemeUninstaller(this);
  3906.     uninstaller.uninstall(aThemeID, isProfile);  
  3907.     
  3908.     // Do this LAST since inferences are made about an item based on
  3909.     // what container it's in.
  3910.     this.removeItemFromContainer(aThemeID, nsIUpdateItem.TYPE_THEME, isProfile);
  3911.   },
  3912.   
  3913.   // Cleans the resource of all its assertionss
  3914.   removeItemMetadata: function nsExtensionsDataSource_removeItemMetadata (aItemID, aItemType)
  3915.   {
  3916.     var item = gRDF.GetResource(getItemPrefix(aItemType) + aItemID);
  3917.     var isProfile = this.isProfileItem(aItemID);
  3918.     var ds = isProfile ? this._profileExtensions : this._appExtensions;
  3919.     
  3920.     var resources = ["targetApplication", "requires"];
  3921.     for (var i = 0; i < resources.length; ++i) {
  3922.       var targetApps = ds.GetTargets(item, this._emR(resources[i]), true);
  3923.       while (targetApps.hasMoreElements()) {
  3924.         var targetApp = targetApps.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3925.         this._cleanResource(targetApp, ds);
  3926.       }
  3927.     }
  3928.  
  3929.     this._cleanResource(item, ds);
  3930.   },
  3931.   
  3932.   _cleanResource: function nsExtensionsDataSource__cleanResource (aResource, aDS)
  3933.   {
  3934.     // Remove outward arcs
  3935.     var arcs = aDS.ArcLabelsOut(aResource);
  3936.     while (arcs.hasMoreElements()) {
  3937.       var arc = arcs.getNext().QueryInterface(Components.interfaces.nsIRDFResource);
  3938.       var value = aDS.GetTarget(aResource, arc, true);
  3939.       if (value)
  3940.         aDS.Unassert(aResource, arc, value);
  3941.     }
  3942.   },
  3943.   
  3944.   moveTop: function nsExtensionsDataSource_moveTop (aItemID)
  3945.   {
  3946.     var extensions = gRDF.GetResource("urn:mozilla:extension:root");
  3947.     var item = this._getResourceForItem(aItemID);
  3948.     var ds = this._getTargetDSFromSource(item);
  3949.     var container = Components.classes["@mozilla.org/rdf/container;1"]
  3950.                               .createInstance(Components.interfaces.nsIRDFContainer);
  3951.     container.Init(ds, extensions);
  3952.     
  3953.     var index = container.IndexOf(item);
  3954.     if (index > 1) {
  3955.       container.RemoveElement(item, false);
  3956.       container.InsertElementAt(item, 1, true);
  3957.     }
  3958.     this._flush(this.isProfileItem(aItemID));
  3959.   },
  3960.   
  3961.   moveUp: function nsExtensionsDataSource_moveUp (aItemID)
  3962.   {
  3963.     var extensions = gRDF.GetResource("urn:mozilla:extension:root");
  3964.     var item = this._getResourceForItem(aItemID);
  3965.     var ds = this._getTargetDSFromSource(item);
  3966.     var container = Components.classes["@mozilla.org/rdf/container;1"]
  3967.                               .createInstance(Components.interfaces.nsIRDFContainer);
  3968.     container.Init(ds, extensions);
  3969.     
  3970.     var item = this._getResourceForItem(aItemID);
  3971.     var index = container.IndexOf(item);
  3972.     if (index > 1) {
  3973.       container.RemoveElement(item, false);
  3974.       container.InsertElementAt(item, index - 1, true);
  3975.     }
  3976.     this._flush(this.isProfileItem(aItemID));
  3977.   },
  3978.   
  3979.   moveDown: function nsExtensionsDataSource_moveDown (aItemID)
  3980.   {
  3981.     var extensions = gRDF.GetResource("urn:mozilla:extension:root");
  3982.     var item = this._getResourceForItem(aItemID);
  3983.     var ds = this._getTargetDSFromSource(item);
  3984.     var container = Components.classes["@mozilla.org/rdf/container;1"]
  3985.                               .createInstance(Components.interfaces.nsIRDFContainer);
  3986.     container.Init(ds, extensions);
  3987.     
  3988.     var item = this._getResourceForItem(aItemID);
  3989.     var index = container.IndexOf(item);
  3990.     var count = container.GetCount();
  3991.     if (index < count) {
  3992.       container.RemoveElement(item, true);
  3993.       container.InsertElementAt(item, index + 1, true);
  3994.     }
  3995.     this._flush(this.isProfileItem(aItemID));
  3996.   },
  3997.   
  3998.   isDownloadItem: function nsExtensionsDataSource_isDownloadItem (aItemID)
  3999.   {
  4000.     return this.getItemProperty(aItemID, "downloadURL") != "";
  4001.   },
  4002.  
  4003.   addDownload: function nsExtensionsDataSource_addDownload (aName, aURL, aIconURL, aItemType)
  4004.   {
  4005.     var root = gRDF.GetResource(getItemRoot(aItemType));
  4006.     
  4007.     var res = gRDF.GetResource(aURL);
  4008.     this._setProperty(this._profileExtensions, res, 
  4009.                       this._emR("name"),
  4010.                       gRDF.GetLiteral(aName))
  4011.     this._setProperty(this._profileExtensions, res, 
  4012.                       this._emR("version"),
  4013.                       gRDF.GetLiteral(" "));
  4014.     this._setProperty(this._profileExtensions, res, 
  4015.                       this._emR("iconURL"),
  4016.                       gRDF.GetLiteral(aIconURL));
  4017.     this._setProperty(this._profileExtensions, res, 
  4018.                       this._emR("downloadURL"),
  4019.                       gRDF.GetLiteral(aURL));
  4020.  
  4021.     var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  4022.                         .createInstance(Components.interfaces.nsIRDFContainer);
  4023.     ctr.Init(this._profileExtensions, root);
  4024.     if (ctr.IndexOf(res) == -1)
  4025.       ctr.InsertElementAt(res, 1, true);
  4026.     
  4027.     this._flush(true);
  4028.   },
  4029.   
  4030.   removeDownload: function nsExtensionsDataSource_removeDownload (aURL, aItemType)
  4031.   {
  4032.     var root = gRDF.GetResource(getItemRoot(aItemType));
  4033.     var res = gRDF.GetResource(aURL);
  4034.     var ctr = Components.classes["@mozilla.org/rdf/container;1"]
  4035.                         .createInstance(Components.interfaces.nsIRDFContainer);
  4036.     ctr.Init(this._profileExtensions, root);
  4037.     ctr.RemoveElement(res, true);
  4038.     this._cleanResource(res, this._profileExtensions);
  4039.     
  4040.     this._flush(true);
  4041.   },
  4042.     
  4043.   flushProgressInfo: function nsExtensionsDataSource_flushProgressInfo (aData)
  4044.   {
  4045.     for (var url in aData) {
  4046.       var res = gRDF.GetResource(url);
  4047.       this._setProperty(this._profileExtensions, res, 
  4048.                         this._emR("state"),
  4049.                         gRDF.GetIntLiteral(aData[url].state));
  4050.       this._setProperty(this._profileExtensions, res, 
  4051.                         this._emR("progress"),
  4052.                         gRDF.GetIntLiteral(aData[url].progress));
  4053.     }
  4054.     this._flush(true);
  4055.   },   
  4056.    
  4057.   loadExtensions: function nsExtensionsDataSource_loadExtensions (aProfile)
  4058.   {
  4059.     var extensionsFile  = getFile(getDirKey(aProfile), 
  4060.                                   [DIR_EXTENSIONS, FILE_EXTENSIONS]);
  4061.     ensureExtensionsFiles(aProfile);
  4062.  
  4063.     var ds = gRDF.GetDataSourceBlocking(getURLSpecFromFile(extensionsFile));
  4064.     if (aProfile) {
  4065.       this._profileExtensions = ds;
  4066.       if (!this._composite) 
  4067.         this._composite = Components.classes["@mozilla.org/rdf/datasource;1?name=composite-datasource"]
  4068.                                     .createInstance(Components.interfaces.nsIRDFDataSource);
  4069.       if (this._appExtensions)
  4070.         this._composite.RemoveDataSource(this._appExtensions);
  4071.       this._composite.AddDataSource(this._profileExtensions);
  4072.       if (this._appExtensions)
  4073.         this._composite.AddDataSource(this._appExtensions);  
  4074.     }
  4075.     else {
  4076.       this._appExtensions = ds;
  4077.       
  4078.       if (!this._composite)
  4079.         this._composite = Components.classes["@mozilla.org/rdf/datasource;1?name=composite-datasource"]
  4080.                                     .createInstance(Components.interfaces.nsIRDFCompositeDataSource);
  4081.       this._composite.AddDataSource(this._appExtensions);
  4082.     }
  4083.   },
  4084.   
  4085.   /////////////////////////////////////////////////////////////////////////////
  4086.   // nsIRDFDataSource
  4087.   get URI()
  4088.   {
  4089.     return "rdf:extensions";
  4090.   },
  4091.   
  4092.   GetSource: function nsExtensionsDataSource_GetSource (aProperty, aTarget, aTruthValue)
  4093.   {
  4094.     return this._composite.GetSource(aProperty, aTarget, aTruthValue);
  4095.   },
  4096.   
  4097.   GetSources: function nsExtensionsDataSource_GetSources (aProperty, aTarget, aTruthValue)
  4098.   {
  4099.     return this._composite.GetSources(aProperty, aTarget, aTruthValue);
  4100.   },
  4101.   
  4102.   _getThemeJARURL: function nsExtensionsDataSource__getThemeJARURL (aSource, aFileName, aFallbackURL)
  4103.   {
  4104.     var id = stripPrefix(aSource.Value, nsIUpdateItem.TYPE_THEME);
  4105.     var chromeDir = getDir(this.isProfileItem(id) ? KEY_PROFILEDIR : KEY_APPDIR, 
  4106.                             [DIR_EXTENSIONS, id, DIR_CHROME]);
  4107.  
  4108.     var jarFile = null;
  4109.     // XXXben hack for pre-configured classic.jar
  4110.     if ((!chromeDir.exists() || !chromeDir.directoryEntries.hasMoreElements()) &&
  4111.         aSource.EqualsNode(gRDF.GetResource("urn:mozilla:theme:{972ce4c6-7e08-4474-a285-3208198ce6fd}")))
  4112.       jarFile = getFile(KEY_APPDIR, ["chrome", "classic.jar"]);
  4113.     if (chromeDir.directoryEntries.hasMoreElements() || jarFile) {
  4114.       if (!jarFile)
  4115.         jarFile = chromeDir.directoryEntries.getNext().QueryInterface(Components.interfaces.nsIFile);
  4116.  
  4117.       if (jarFile.exists()) {
  4118.         var zipReader = Components.classes["@mozilla.org/libjar/zip-reader;1"]
  4119.                                   .createInstance(Components.interfaces.nsIZipReader);
  4120.         zipReader.init(jarFile);
  4121.         zipReader.open();
  4122.         var url = aFallbackURL;
  4123.         try {
  4124.           zipReader.getEntry(aFileName);
  4125.           url = "jar:" + getURLSpecFromFile(jarFile) + "!/" + aFileName; 
  4126.         }
  4127.         catch (e) { }
  4128.         zipReader.close();
  4129.         
  4130.         if (url)
  4131.           return gRDF.GetResource(url);
  4132.       }
  4133.     }
  4134.     return null;
  4135.   },
  4136.   
  4137.   GetTarget: function nsExtensionsDataSource_GetTarget(aSource, aProperty, aTruthValue)
  4138.   {
  4139.     if (!aSource)
  4140.       return null;
  4141.       
  4142.     if (aProperty.EqualsNode(this._emR("iconURL"))) {
  4143.       var itemType = getItemType(aSource.Value);
  4144.       if (itemType != -1 && itemType & nsIUpdateItem.TYPE_EXTENSION) {
  4145.         var hasIconURL = this._composite.hasArcOut(aSource, aProperty);
  4146.         // If the download entry doesn't have a IconURL property, use a
  4147.         // generic icon URL instead.
  4148.         if (!hasIconURL)
  4149.           return gRDF.GetResource("chrome://mozapps/skin/xpinstall/xpinstallItemGeneric.png");
  4150.         else {
  4151.           var iconURL = this._composite.GetTarget(aSource, aProperty, true);
  4152.           iconURL = iconURL.QueryInterface(Components.interfaces.nsIRDFLiteral).Value;
  4153.           var cr = Components.classes["@mozilla.org/chrome/chrome-registry;1"]
  4154.                              .getService(Components.interfaces.nsIChromeRegistry);
  4155.           var ioServ = Components.classes["@mozilla.org/network/io-service;1"]
  4156.                                  .getService(Components.interfaces.nsIIOService);
  4157.           var uri = ioServ.newURI(iconURL, null, null);
  4158.           try {
  4159.             cr.convertChromeURL(uri);
  4160.           }
  4161.           catch(e) {
  4162.             // bogus URI, supply a generic icon. 
  4163.             return gRDF.GetResource("chrome://mozapps/skin/xpinstall/xpinstallItemGeneric.png");
  4164.           }
  4165.         }
  4166.       }
  4167.       else if (itemType != -1 && itemType & nsIUpdateItem.TYPE_THEME) {
  4168.         var res = this._getThemeJARURL(aSource, "icon.png", "chrome://mozapps/skin/extensions/themeGeneric.png");
  4169.         if (res)
  4170.           return res;
  4171.       }
  4172.     }
  4173.     else if (aProperty.EqualsNode(this._emR("previewImage"))) {
  4174.       var itemType = getItemType(aSource.Value);
  4175.       if (itemType != -1 && itemType & nsIUpdateItem.TYPE_THEME) {
  4176.         var res = this._getThemeJARURL(aSource, "preview.png", null);
  4177.         if (res)
  4178.           return res;
  4179.       }
  4180.     }
  4181.     else if (aProperty.EqualsNode(this._emR("installLocation"))) {
  4182.       var arcs = this._profileExtensions.ArcLabelsOut(aSource);
  4183.       return arcs.hasMoreElements() ? this._emL("profile") : this._emL("global");
  4184.     }
  4185.     else if (aProperty.EqualsNode(this._emR("disabled"))) {
  4186.       if (this.safeMode) 
  4187.         return this._emL("true");
  4188.       // fall through to default.
  4189.     }
  4190.     else if (aProperty.EqualsNode(this._emR("itemType"))) {
  4191.       // We can try and infer the type from presence in one of the 
  4192.       // item lists.
  4193.       var rdfc = Components.classes["@mozilla.org/rdf/container;1"]
  4194.                           .createInstance(Components.interfaces.nsIRDFContainer);
  4195.       rdfc.Init(this, gRDF.GetResource(ROOT_EXTENSION));
  4196.       if (rdfc.IndexOf(aSource) != -1) 
  4197.         return this._emL("extension");
  4198.     
  4199.       rdfc.Init(this, gRDF.GetResource(ROOT_THEME));
  4200.       if (rdfc.IndexOf(aSource) != -1) 
  4201.         return this._emL("theme");
  4202.     }
  4203.     else if (aProperty.EqualsNode(this._emR("compatible"))) {
  4204.       var type = getItemType(aSource.Value);
  4205.       var id = stripPrefix(aSource.Value, type);
  4206.       var targetAppInfo = this.getTargetApplicationInfo(id, this, type);
  4207.       if (!targetAppInfo)
  4208.         return this._emL("false");
  4209.       getVersionChecker();
  4210.       
  4211.       var appVersion = gPref.getCharPref(PREF_EM_APP_EXTENSIONS_VERSION);
  4212.       if (gVersionChecker.compare(targetAppInfo.maxVersion, appVersion) < 0 || 
  4213.           gVersionChecker.compare(appVersion, targetAppInfo.minVersion) < 0) {
  4214.         // OK, this item is incompatible. 
  4215.         return this._emL("false");
  4216.       }
  4217.       return this._emL("true");
  4218.     }
  4219.     else if (aProperty.EqualsNode(this._emR("displayDescription"))) {
  4220.       // We have a separate property for the description of the extension items
  4221.       // which is displayed in the EM list - because we overload this value with
  4222.       // alternative messages when the extension is disabled because of 
  4223.       // incompatibility.
  4224.       var disabled = this.getItemProperty(stripPrefix(aSource.Value, getItemType(aSource.Value)), 
  4225.                                           "disabled");
  4226.       if (disabled == "true") {
  4227.         // See if this item was disabled because it was incompatible. 
  4228.         // XXXben potential visual-glitch bug here with extensions whose install.rdf 
  4229.         //        manifests state that they are incompatible but when phone home checking
  4230.         //        reveals that they are compatible and they are installed the 
  4231.         //        incompatible metadata is written anyway and will remain in the ds
  4232.         //        until the next background update check corrects it - this means that
  4233.         //        when a compatible extension is installed in this manner it is 
  4234.         //        likely that when it is disabled it will show this special-case
  4235.         //        error message.
  4236.         var compatible = this.getItemProperty(stripPrefix(aSource.Value, getItemType(aSource.Value)), 
  4237.                                               "compatible");
  4238.         if (compatible != "true") {
  4239.           var sbs = Components.classes["@mozilla.org/intl/stringbundle;1"]
  4240.                               .getService(Components.interfaces.nsIStringBundleService);
  4241.           var extensionStrings = sbs.createBundle("chrome://mozapps/locale/extensions/extensions.properties");
  4242.           var brandStrings = sbs.createBundle("chrome://global/locale/brand.properties");
  4243.           var brandShortName = brandStrings.GetStringFromName("brandShortName");
  4244.           var appVersion = gPref.getCharPref(PREF_EM_APP_VERSION);
  4245.           var incompatibleMessage = extensionStrings.formatStringFromName("incompatibleExtension", 
  4246.                                                                           [brandShortName, appVersion], 2);
  4247.           return this._emL(incompatibleMessage);
  4248.         }
  4249.       }
  4250.       // Use the "description" property. 
  4251.       return this.GetTarget(aSource, this._emR("description"), aTruthValue);
  4252.     }
  4253.     else if (aProperty.EqualsNode(this._emR("name")) || 
  4254.              aProperty.EqualsNode(this._emR("description")) || 
  4255.              aProperty.EqualsNode(this._emR("creator")) || 
  4256.              aProperty.EqualsNode(this._emR("homepageURL"))) {
  4257.       // These are localizable properties that a language pack supplied by the 
  4258.       // Extension may override.          
  4259.       var prefName = PREF_EM_EXTENSION_FORMAT.replace(/%UUID%/, 
  4260.                                                       stripPrefix(aSource.Value, 
  4261.                                                                   nsIUpdateItem.TYPE_EXTENSION)) + 
  4262.                       stripPropertyPrefix(aProperty.Value, EM_NS_PREFIX);
  4263.       try {
  4264.         var value = gPref.getComplexValue(prefName, 
  4265.                                           Components.interfaces.nsIPrefLocalizedString);
  4266.         if (value.data) 
  4267.           return this._emL(value.data);
  4268.       }
  4269.       catch (e) {
  4270.       }
  4271.     }
  4272.     
  4273.     return this._composite.GetTarget(aSource, aProperty, aTruthValue);
  4274.   },
  4275.   
  4276.   GetTargets: function nsExtensionsDataSource_GetTargets (aSource, aProperty, aTruthValue)
  4277.   {
  4278.     if (aProperty.EqualsNode(this._emR("name")) ||
  4279.         aProperty.EqualsNode(this._emR("contributor"))) {
  4280.       // These are localizable properties that a language pack supplied by the 
  4281.       // Extension may override.          
  4282.       var contributors = [];
  4283.       var prefName = PREF_EM_EXTENSION_FORMAT.replace(/%UUID%/, 
  4284.                                                       stripPrefix(aSource.Value, 
  4285.                                                                   nsIUpdateItem.TYPE_EXTENSION)) + 
  4286.                       stripPropertyPrefix(aProperty.Value, EM_NS_PREFIX);
  4287.       var i = 0;
  4288.       do {
  4289.         try {
  4290.           var value = gPref.getComplexValue(prefName + "." + ++i, 
  4291.                                             Components.interfaces.nsIPrefLocalizedString);
  4292.           if (value.data) 
  4293.             contributors.push(this._emL(value.data));
  4294.         }
  4295.         catch (e) {
  4296.           try {
  4297.             var value = gPref.getComplexValue(prefName, 
  4298.                                               Components.interfaces.nsIPrefLocalizedString);
  4299.             if (value.data) 
  4300.               contributors.push(this._emL(value.data));
  4301.           }
  4302.           catch (e) {
  4303.           }
  4304.           break;
  4305.         }
  4306.       }
  4307.       while (1);
  4308.       if (contributors.length > 0)
  4309.         return new ArrayEnumerator(contributors);
  4310.     }
  4311.     return this._composite.GetTargets(aSource, aProperty, aTruthValue);
  4312.   },
  4313.   
  4314.   _getTargetDSFromSource: function nsExtensionsDataSource__getTargetDSFromSource (aSource)
  4315.   {
  4316.     var itemID = stripPrefix(aSource.Value, nsIUpdateItem.TYPE_ADDON);
  4317.     return this.isProfileItem(itemID) ? this._profileExtensions : this._appExtensions;
  4318.   },
  4319.   
  4320.   Assert: function nsExtensionsDataSource_Assert (aSource, aProperty, aTarget, aTruthValue)
  4321.   {
  4322.     var targetDS = this._getTargetDSFromSource(aSource);
  4323.     targetDS.Assert(aSource, aProperty, aTarget, aTruthValue);
  4324.   },
  4325.   
  4326.   Unassert: function nsExtensionsDataSource_Unassert (aSource, aProperty, aTarget)
  4327.   {
  4328.     var targetDS = this._getTargetDSFromSource(aSource);
  4329.     targetDS.Unassert(aSource, aProperty, aTarget);
  4330.   },
  4331.   
  4332.   Change: function nsExtensionsDataSource_Change (aSource, aProperty, aOldTarget, aNewTarget)
  4333.   {
  4334.     var targetDS = this._getTargetDSFromSource(aSource);
  4335.     targetDS.Change(aSource, aProperty, aOldTarget, aNewTarget);
  4336.   },
  4337.  
  4338.   Move: function nsExtensionsDataSource_Move (aSource, aNewSource, aProperty, aTarget)
  4339.   {
  4340.     var targetDS = this._getTargetDSFromSource(aSource);
  4341.     targetDS.Move(aSource, aNewSource, aProperty, aTarget);
  4342.   },
  4343.   
  4344.   HasAssertion: function nsExtensionsDataSource_HasAssertion (aSource, aProperty, aTarget, aTruthValue)
  4345.   {
  4346.     if (!aSource || !aProperty || !aTarget)
  4347.       return false;
  4348.     return this._composite.HasAssertion(aSource, aProperty, aTarget, aTruthValue);
  4349.   },
  4350.   
  4351.   _observers: [],
  4352.   AddObserver: function nsExtensionsDataSource_AddObserver (aObserver)
  4353.   {
  4354.     for (var i = 0; i < this._observers.length; ++i) {
  4355.       if (this._observers[i] == aObserver) 
  4356.         return;
  4357.     }
  4358.     this._observers.push(aObserver);
  4359.     this._composite.AddObserver(aObserver);
  4360.   },
  4361.   
  4362.   RemoveObserver: function nsExtensionsDataSource_RemoveObserver (aObserver)
  4363.   {
  4364.     for (var i = 0; i < this._observers.length; ++i) {
  4365.       if (this._observers[i] == aObserver) 
  4366.         this._observers.splice(i, 1);
  4367.     }
  4368.     this._composite.RemoveObserver(aObserver);
  4369.   },
  4370.   
  4371.   ArcLabelsIn: function nsExtensionsDataSource_ArcLabelsIn (aNode)
  4372.   {
  4373.     return this._composite.ArcLabelsIn(aNode);
  4374.   },
  4375.   
  4376.   ArcLabelsOut: function nsExtensionsDataSource_ArcLabelsOut (aSource)
  4377.   {
  4378.     return this._composite.ArcLabelsOut(aSource);
  4379.   },
  4380.   
  4381.   GetAllResources: function nsExtensionsDataSource_GetAllResources ()
  4382.   {
  4383.     return this._composite.GetAllResources();
  4384.   },
  4385.   
  4386.   IsCommandEnabled: function nsExtensionsDataSource_IsCommandEnabled (aSources, aCommand, aArguments)
  4387.   {
  4388.     return this._composite.IsCommandEnabled(aSources, aCommand, aArguments);
  4389.   },
  4390.   
  4391.   DoCommand: function nsExtensionsDataSource_DoCommand (aSources, aCommand, aArguments)
  4392.   {
  4393.     this._composite.DoCommand(aSources, aCommand, aArguments);
  4394.   },
  4395.   
  4396.   GetAllCmds: function nsExtensionsDataSource_GetAllCmds (aSource)
  4397.   {
  4398.     return this._composite.GetAllCmds(aSource);
  4399.   },
  4400.   
  4401.   hasArcIn: function nsExtensionsDataSource_hasArcIn (aNode, aArc)
  4402.   {
  4403.     return this._composite.hasArcIn(aNode, aArc);
  4404.   },
  4405.   
  4406.   hasArcOut: function nsExtensionsDataSource_hasArcOut (aSource, aArc)
  4407.   {
  4408.     return this._composite.hasArcOut(aSource, aArc);
  4409.   },
  4410.   
  4411.   beginUpdateBatch: function nsExtensionsDataSource_beginUpdateBatch ()
  4412.   {
  4413.     return this._composite.beginUpdateBatch();
  4414.   },
  4415.   
  4416.   endUpdateBatch: function nsExtensionsDataSource_endUpdateBatch ()
  4417.   {
  4418.     return this._composite.endUpdateBatch();
  4419.   },
  4420.   
  4421.   /////////////////////////////////////////////////////////////////////////////
  4422.   // nsIRDFRemoteDataSource
  4423.   
  4424.   get loaded()
  4425.   {
  4426.     throw Components.results.NS_ERROR_NOT_IMPLEMENTED;
  4427.   },
  4428.   
  4429.   Init: function nsExtensionsDataSource_Init (aURI)
  4430.   {
  4431.   },
  4432.   
  4433.   Refresh: function nsExtensionsDataSource_Refresh (aBlocking)
  4434.   {
  4435.   },
  4436.   
  4437.   Flush: function nsExtensionsDataSource_Flush ()
  4438.   {
  4439.     this._flush(false);
  4440.     this._flush(true);
  4441.   },
  4442.   
  4443.   FlushTo: function nsExtensionsDataSource_FlushTo (aURI)
  4444.   {
  4445.   },
  4446.   
  4447.   _flush: function nsExtensionsDataSource__flush (aIsProfile)
  4448.   { 
  4449.     var ds = aIsProfile ? this._profileExtensions : this._appExtensions;
  4450.     var rds = ds.QueryInterface(Components.interfaces.nsIRDFRemoteDataSource);
  4451.     rds.Flush();
  4452.   },
  4453.  
  4454.   /////////////////////////////////////////////////////////////////////////////
  4455.   // nsISupports
  4456.   QueryInterface: function nsExtensionsDataSource_QueryInterface (aIID) 
  4457.   {
  4458.     if (!aIID.equals(Components.interfaces.nsIRDFDataSource) &&
  4459.         !aIID.equals(Components.interfaces.nsIRDFRemoteDataSource) && 
  4460.         !aIID.equals(Components.interfaces.nsISupports))
  4461.       throw Components.results.NS_ERROR_NO_INTERFACE;
  4462.     return this;
  4463.   }
  4464. };
  4465.  
  4466.  
  4467. var gModule = {
  4468.   _firstTime: true,
  4469.   
  4470.   registerSelf: function (aComponentManager, aFileSpec, aLocation, aType) 
  4471.   {
  4472.     if (this._firstTime) {
  4473.       this._firstTime = false;
  4474.       throw Components.results.NS_ERROR_FACTORY_REGISTER_AGAIN;
  4475.     }
  4476.     aComponentManager = aComponentManager.QueryInterface(Components.interfaces.nsIComponentRegistrar);
  4477.     
  4478.     for (var key in this._objects) {
  4479.       var obj = this._objects[key];
  4480.       aComponentManager.registerFactoryLocation(obj.CID, obj.className, obj.contractID,
  4481.                                                 aFileSpec, aLocation, aType);
  4482.     }
  4483.  
  4484. /*    
  4485.     // Make the Extension Manager a startup observer
  4486.     var categoryManager = Components.classes["@mozilla.org/categorymanager;1"]
  4487.                                     .getService(Components.interfaces.nsICategoryManager);
  4488.     categoryManager.addCategoryEntry("app-startup", this._objects.manager.className,
  4489.                                      "service," + this._objects.manager.contractID, 
  4490.                                      true, true, null);
  4491.  */
  4492.   },
  4493.   
  4494.   getClassObject: function (aComponentManager, aCID, aIID) 
  4495.   {
  4496.     if (!aIID.equals(Components.interfaces.nsIFactory))
  4497.       throw Components.results.NS_ERROR_NOT_IMPLEMENTED;
  4498.  
  4499.     for (var key in this._objects) {
  4500.       if (aCID.equals(this._objects[key].CID))
  4501.         return this._objects[key].factory;
  4502.     }
  4503.     
  4504.     throw Components.results.NS_ERROR_NO_INTERFACE;
  4505.   },
  4506.   
  4507.   _objects: {
  4508.     manager: { CID        : nsExtensionManager.prototype.classID,
  4509.                contractID : nsExtensionManager.prototype.contractID,
  4510.                className  : nsExtensionManager.prototype.classDescription,
  4511.                factory    : {
  4512.                               createInstance: function (aOuter, aIID) 
  4513.                               {
  4514.                                 if (aOuter != null)
  4515.                                   throw Components.results.NS_ERROR_NO_AGGREGATION;
  4516.                                 
  4517.                                 return (new nsExtensionManager()).QueryInterface(aIID);
  4518.                               }
  4519.                             }
  4520.              }
  4521.    },
  4522.   
  4523.   canUnload: function (aComponentManager) 
  4524.   {
  4525.     return true;
  4526.   }
  4527. };
  4528.  
  4529. function NSGetModule(compMgr, fileSpec) 
  4530. {
  4531.   return gModule;
  4532. }
  4533.