forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			617 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			617 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| var EXPORTED_SYMBOLS = ["DistributionCustomizer"];
 | |
| 
 | |
| const DISTRIBUTION_CUSTOMIZATION_COMPLETE_TOPIC =
 | |
|   "distribution-customization-complete";
 | |
| 
 | |
| const PREF_CACHED_FILE_EXISTENCE = "distribution.iniFile.exists.value";
 | |
| const PREF_CACHED_FILE_APPVERSION = "distribution.iniFile.exists.appversion";
 | |
| 
 | |
| const { AppConstants } = ChromeUtils.import(
 | |
|   "resource://gre/modules/AppConstants.jsm"
 | |
| );
 | |
| const { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
 | |
| ChromeUtils.defineModuleGetter(
 | |
|   this,
 | |
|   "Preferences",
 | |
|   "resource://gre/modules/Preferences.jsm"
 | |
| );
 | |
| ChromeUtils.defineModuleGetter(
 | |
|   this,
 | |
|   "PlacesUtils",
 | |
|   "resource://gre/modules/PlacesUtils.jsm"
 | |
| );
 | |
| 
 | |
| function DistributionCustomizer() {}
 | |
| 
 | |
| DistributionCustomizer.prototype = {
 | |
|   // These prefixes must only contain characters
 | |
|   // allowed by PlacesUtils.isValidGuid
 | |
|   BOOKMARK_GUID_PREFIX: "DstB-",
 | |
|   FOLDER_GUID_PREFIX: "DstF-",
 | |
| 
 | |
|   get _iniFile() {
 | |
|     // For parallel xpcshell testing purposes allow loading the distribution.ini
 | |
|     // file from the profile folder through an hidden pref.
 | |
|     let loadFromProfile = Services.prefs.getBoolPref(
 | |
|       "distribution.testing.loadFromProfile",
 | |
|       false
 | |
|     );
 | |
| 
 | |
|     let iniFile;
 | |
|     try {
 | |
|       iniFile = loadFromProfile
 | |
|         ? Services.dirsvc.get("ProfD", Ci.nsIFile)
 | |
|         : Services.dirsvc.get("XREAppDist", Ci.nsIFile);
 | |
|       if (loadFromProfile) {
 | |
|         iniFile.leafName = "distribution";
 | |
|       }
 | |
|       iniFile.append("distribution.ini");
 | |
|     } catch (ex) {}
 | |
| 
 | |
|     this.__defineGetter__("_iniFile", () => iniFile);
 | |
|     return iniFile;
 | |
|   },
 | |
| 
 | |
|   get _hasDistributionIni() {
 | |
|     if (Services.prefs.prefHasUserValue(PREF_CACHED_FILE_EXISTENCE)) {
 | |
|       let knownForVersion = Services.prefs.getStringPref(
 | |
|         PREF_CACHED_FILE_APPVERSION,
 | |
|         "unknown"
 | |
|       );
 | |
|       if (knownForVersion == AppConstants.MOZ_APP_VERSION) {
 | |
|         return Services.prefs.getBoolPref(PREF_CACHED_FILE_EXISTENCE);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     let fileExists = this._iniFile.exists();
 | |
|     Services.prefs.setBoolPref(PREF_CACHED_FILE_EXISTENCE, fileExists);
 | |
|     Services.prefs.setStringPref(
 | |
|       PREF_CACHED_FILE_APPVERSION,
 | |
|       AppConstants.MOZ_APP_VERSION
 | |
|     );
 | |
| 
 | |
|     this.__defineGetter__("_hasDistributionIni", () => fileExists);
 | |
|     return fileExists;
 | |
|   },
 | |
| 
 | |
|   get _ini() {
 | |
|     let ini = null;
 | |
|     try {
 | |
|       if (this._hasDistributionIni) {
 | |
|         ini = Cc["@mozilla.org/xpcom/ini-parser-factory;1"]
 | |
|           .getService(Ci.nsIINIParserFactory)
 | |
|           .createINIParser(this._iniFile);
 | |
|       }
 | |
|     } catch (e) {
 | |
|       if (e.result == Cr.NS_ERROR_FILE_NOT_FOUND) {
 | |
|         // We probably had cached the file existence as true,
 | |
|         // but it no longer exists. We could set the new cache
 | |
|         // value here, but let's just invalidate the cache and
 | |
|         // let it be cached by a single code path on the next check.
 | |
|         Services.prefs.clearUserPref(PREF_CACHED_FILE_EXISTENCE);
 | |
|       } else {
 | |
|         // Unable to parse INI.
 | |
|         Cu.reportError("Unable to parse distribution.ini");
 | |
|       }
 | |
|     }
 | |
|     this.__defineGetter__("_ini", () => ini);
 | |
|     return this._ini;
 | |
|   },
 | |
| 
 | |
|   get _locale() {
 | |
|     const locale = Services.locale.requestedLocale || "en-US";
 | |
|     this.__defineGetter__("_locale", () => locale);
 | |
|     return this._locale;
 | |
|   },
 | |
| 
 | |
|   get _language() {
 | |
|     let language = this._locale.split("-")[0];
 | |
|     this.__defineGetter__("_language", () => language);
 | |
|     return this._language;
 | |
|   },
 | |
| 
 | |
|   async _parseBookmarksSection(parentGuid, section) {
 | |
|     let keys = Array.from(this._ini.getKeys(section)).sort();
 | |
|     let re = /^item\.(\d+)\.(\w+)\.?(\w*)/;
 | |
|     let items = {};
 | |
|     let defaultIndex = -1;
 | |
|     let maxIndex = -1;
 | |
| 
 | |
|     for (let key of keys) {
 | |
|       let m = re.exec(key);
 | |
|       if (m) {
 | |
|         let [, itemIndex, iprop, ilocale] = m;
 | |
|         itemIndex = parseInt(itemIndex);
 | |
| 
 | |
|         if (ilocale) {
 | |
|           continue;
 | |
|         }
 | |
| 
 | |
|         if (keys.includes(key + "." + this._locale)) {
 | |
|           key += "." + this._locale;
 | |
|         } else if (keys.includes(key + "." + this._language)) {
 | |
|           key += "." + this._language;
 | |
|         }
 | |
| 
 | |
|         if (!items[itemIndex]) {
 | |
|           items[itemIndex] = {};
 | |
|         }
 | |
|         items[itemIndex][iprop] = this._ini.getString(section, key);
 | |
| 
 | |
|         if (iprop == "type" && items[itemIndex].type == "default") {
 | |
|           defaultIndex = itemIndex;
 | |
|         }
 | |
| 
 | |
|         if (maxIndex < itemIndex) {
 | |
|           maxIndex = itemIndex;
 | |
|         }
 | |
|       } else {
 | |
|         dump(`Key did not match: ${key}\n`);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     let prependIndex = 0;
 | |
|     for (let itemIndex = 0; itemIndex <= maxIndex; itemIndex++) {
 | |
|       if (!items[itemIndex]) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       let index = PlacesUtils.bookmarks.DEFAULT_INDEX;
 | |
|       let item = items[itemIndex];
 | |
| 
 | |
|       switch (item.type) {
 | |
|         case "default":
 | |
|           break;
 | |
| 
 | |
|         case "folder":
 | |
|           if (itemIndex < defaultIndex) {
 | |
|             index = prependIndex++;
 | |
|           }
 | |
| 
 | |
|           let folder = await PlacesUtils.bookmarks.insert({
 | |
|             type: PlacesUtils.bookmarks.TYPE_FOLDER,
 | |
|             guid: PlacesUtils.generateGuidWithPrefix(this.FOLDER_GUID_PREFIX),
 | |
|             parentGuid,
 | |
|             index,
 | |
|             title: item.title,
 | |
|           });
 | |
| 
 | |
|           await this._parseBookmarksSection(
 | |
|             folder.guid,
 | |
|             "BookmarksFolder-" + item.folderId
 | |
|           );
 | |
|           break;
 | |
| 
 | |
|         case "separator":
 | |
|           if (itemIndex < defaultIndex) {
 | |
|             index = prependIndex++;
 | |
|           }
 | |
| 
 | |
|           await PlacesUtils.bookmarks.insert({
 | |
|             type: PlacesUtils.bookmarks.TYPE_SEPARATOR,
 | |
|             parentGuid,
 | |
|             index,
 | |
|           });
 | |
|           break;
 | |
| 
 | |
|         case "livemark":
 | |
|           // Livemarks are no more supported, instead of a livemark we'll insert
 | |
|           // a bookmark pointing to the site uri, if available.
 | |
|           if (!item.siteLink) {
 | |
|             break;
 | |
|           }
 | |
|           if (itemIndex < defaultIndex) {
 | |
|             index = prependIndex++;
 | |
|           }
 | |
| 
 | |
|           await PlacesUtils.bookmarks.insert({
 | |
|             parentGuid,
 | |
|             index,
 | |
|             title: item.title,
 | |
|             url: item.siteLink,
 | |
|           });
 | |
|           break;
 | |
| 
 | |
|         case "bookmark":
 | |
|         default:
 | |
|           if (itemIndex < defaultIndex) {
 | |
|             index = prependIndex++;
 | |
|           }
 | |
| 
 | |
|           await PlacesUtils.bookmarks.insert({
 | |
|             guid: PlacesUtils.generateGuidWithPrefix(this.BOOKMARK_GUID_PREFIX),
 | |
|             parentGuid,
 | |
|             index,
 | |
|             title: item.title,
 | |
|             url: item.link,
 | |
|           });
 | |
| 
 | |
|           if (item.icon && item.iconData) {
 | |
|             try {
 | |
|               let faviconURI = Services.io.newURI(item.icon);
 | |
|               PlacesUtils.favicons.replaceFaviconDataFromDataURL(
 | |
|                 faviconURI,
 | |
|                 item.iconData,
 | |
|                 0,
 | |
|                 Services.scriptSecurityManager.getSystemPrincipal()
 | |
|               );
 | |
| 
 | |
|               PlacesUtils.favicons.setAndFetchFaviconForPage(
 | |
|                 Services.io.newURI(item.link),
 | |
|                 faviconURI,
 | |
|                 false,
 | |
|                 PlacesUtils.favicons.FAVICON_LOAD_NON_PRIVATE,
 | |
|                 null,
 | |
|                 Services.scriptSecurityManager.getSystemPrincipal()
 | |
|               );
 | |
|             } catch (e) {
 | |
|               Cu.reportError(e);
 | |
|             }
 | |
|           }
 | |
| 
 | |
|           break;
 | |
|       }
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   _newProfile: false,
 | |
|   _customizationsApplied: false,
 | |
|   applyCustomizations: function DIST_applyCustomizations() {
 | |
|     this._customizationsApplied = true;
 | |
| 
 | |
|     if (!Services.prefs.prefHasUserValue("browser.migration.version")) {
 | |
|       this._newProfile = true;
 | |
|     }
 | |
| 
 | |
|     if (!this._ini) {
 | |
|       return this._checkCustomizationComplete();
 | |
|     }
 | |
| 
 | |
|     if (!this._prefDefaultsApplied) {
 | |
|       this.applyPrefDefaults();
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   _bookmarksApplied: false,
 | |
|   async applyBookmarks() {
 | |
|     await this._doApplyBookmarks();
 | |
|     this._bookmarksApplied = true;
 | |
|     this._checkCustomizationComplete();
 | |
|   },
 | |
| 
 | |
|   async _doApplyBookmarks() {
 | |
|     if (!this._ini) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     let sections = enumToObject(this._ini.getSections());
 | |
| 
 | |
|     // The global section, and several of its fields, is required
 | |
|     // (we also check here to be consistent with applyPrefDefaults below)
 | |
|     if (!sections.Global) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     let globalPrefs = enumToObject(this._ini.getKeys("Global"));
 | |
|     if (!(globalPrefs.id && globalPrefs.version && globalPrefs.about)) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     let bmProcessedPref;
 | |
|     try {
 | |
|       bmProcessedPref = this._ini.getString(
 | |
|         "Global",
 | |
|         "bookmarks.initialized.pref"
 | |
|       );
 | |
|     } catch (e) {
 | |
|       bmProcessedPref =
 | |
|         "distribution." +
 | |
|         this._ini.getString("Global", "id") +
 | |
|         ".bookmarksProcessed";
 | |
|     }
 | |
| 
 | |
|     if (Services.prefs.getBoolPref(bmProcessedPref, false)) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     let ProfileAge = ChromeUtils.import(
 | |
|       "resource://gre/modules/ProfileAge.jsm",
 | |
|       {}
 | |
|     ).ProfileAge;
 | |
|     let profileAge = await ProfileAge();
 | |
|     let resetDate = await profileAge.reset;
 | |
| 
 | |
|     // If the profile has been reset, don't recreate bookmarks.
 | |
|     if (!resetDate) {
 | |
|       if (sections.BookmarksMenu) {
 | |
|         await this._parseBookmarksSection(
 | |
|           PlacesUtils.bookmarks.menuGuid,
 | |
|           "BookmarksMenu"
 | |
|         );
 | |
|       }
 | |
|       if (sections.BookmarksToolbar) {
 | |
|         await this._parseBookmarksSection(
 | |
|           PlacesUtils.bookmarks.toolbarGuid,
 | |
|           "BookmarksToolbar"
 | |
|         );
 | |
|       }
 | |
|     }
 | |
|     Services.prefs.setBoolPref(bmProcessedPref, true);
 | |
|   },
 | |
| 
 | |
|   _prefDefaultsApplied: false,
 | |
|   applyPrefDefaults: function DIST_applyPrefDefaults() {
 | |
|     this._prefDefaultsApplied = true;
 | |
|     if (!this._ini) {
 | |
|       return this._checkCustomizationComplete();
 | |
|     }
 | |
| 
 | |
|     let sections = enumToObject(this._ini.getSections());
 | |
| 
 | |
|     // The global section, and several of its fields, is required
 | |
|     if (!sections.Global) {
 | |
|       return this._checkCustomizationComplete();
 | |
|     }
 | |
|     let globalPrefs = enumToObject(this._ini.getKeys("Global"));
 | |
|     if (!(globalPrefs.id && globalPrefs.version)) {
 | |
|       return this._checkCustomizationComplete();
 | |
|     }
 | |
|     let distroID = this._ini.getString("Global", "id");
 | |
|     if (!globalPrefs.about && !distroID.startsWith("mozilla-")) {
 | |
|       // About is required unless it is a mozilla distro.
 | |
|       return this._checkCustomizationComplete();
 | |
|     }
 | |
| 
 | |
|     let defaults = new Preferences({ defaultBranch: true });
 | |
| 
 | |
|     // Global really contains info we set as prefs.  They're only
 | |
|     // separate because they are "special" (read: required)
 | |
| 
 | |
|     defaults.set("distribution.id", distroID);
 | |
|     defaults.set(
 | |
|       "distribution.version",
 | |
|       this._ini.getString("Global", "version")
 | |
|     );
 | |
| 
 | |
|     let partnerAbout;
 | |
|     try {
 | |
|       if (globalPrefs["about." + this._locale]) {
 | |
|         partnerAbout = this._ini.getString("Global", "about." + this._locale);
 | |
|       } else if (globalPrefs["about." + this._language]) {
 | |
|         partnerAbout = this._ini.getString("Global", "about." + this._language);
 | |
|       } else {
 | |
|         partnerAbout = this._ini.getString("Global", "about");
 | |
|       }
 | |
|       defaults.set("distribution.about", partnerAbout);
 | |
|     } catch (e) {
 | |
|       /* ignore bad prefs due to bug 895473 and move on */
 | |
|       Cu.reportError(e);
 | |
|     }
 | |
| 
 | |
|     var usedPreferences = [];
 | |
| 
 | |
|     if (sections["Preferences-" + this._locale]) {
 | |
|       for (let key of this._ini.getKeys("Preferences-" + this._locale)) {
 | |
|         try {
 | |
|           let value = this._ini.getString("Preferences-" + this._locale, key);
 | |
|           if (value) {
 | |
|             defaults.set(key, parseValue(value));
 | |
|           }
 | |
|           usedPreferences.push(key);
 | |
|         } catch (e) {
 | |
|           /* ignore bad prefs and move on */
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (sections["Preferences-" + this._language]) {
 | |
|       for (let key of this._ini.getKeys("Preferences-" + this._language)) {
 | |
|         if (usedPreferences.indexOf(key) > -1) {
 | |
|           continue;
 | |
|         }
 | |
|         try {
 | |
|           let value = this._ini.getString("Preferences-" + this._language, key);
 | |
|           if (value) {
 | |
|             defaults.set(key, parseValue(value));
 | |
|           }
 | |
|           usedPreferences.push(key);
 | |
|         } catch (e) {
 | |
|           /* ignore bad prefs and move on */
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (sections.Preferences) {
 | |
|       for (let key of this._ini.getKeys("Preferences")) {
 | |
|         if (usedPreferences.indexOf(key) > -1) {
 | |
|           continue;
 | |
|         }
 | |
|         try {
 | |
|           let value = this._ini.getString("Preferences", key);
 | |
|           if (value) {
 | |
|             value = value.replace(/%LOCALE%/g, this._locale);
 | |
|             value = value.replace(/%LANGUAGE%/g, this._language);
 | |
|             if (key == "general.useragent.locale") {
 | |
|               defaults.set("intl.locale.requested", parseValue(value));
 | |
|             } else {
 | |
|               defaults.set(key, parseValue(value));
 | |
|             }
 | |
|           }
 | |
|         } catch (e) {
 | |
|           /* ignore bad prefs and move on */
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (this._ini.getString("Global", "id") == "yandex") {
 | |
|       // All yandex distributions have the same distribution ID,
 | |
|       // so we're using an internal preference to name them correctly.
 | |
|       // This is needed for search to work properly.
 | |
|       try {
 | |
|         defaults.set(
 | |
|           "distribution.id",
 | |
|           defaults
 | |
|             .get("extensions.yasearch@yandex.ru.clids.vendor")
 | |
|             .replace("firefox", "yandex")
 | |
|         );
 | |
|       } catch (e) {
 | |
|         // Just use the default distribution ID.
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     let localizedStr = Cc["@mozilla.org/pref-localizedstring;1"].createInstance(
 | |
|       Ci.nsIPrefLocalizedString
 | |
|     );
 | |
| 
 | |
|     var usedLocalizablePreferences = [];
 | |
| 
 | |
|     if (sections["LocalizablePreferences-" + this._locale]) {
 | |
|       for (let key of this._ini.getKeys(
 | |
|         "LocalizablePreferences-" + this._locale
 | |
|       )) {
 | |
|         try {
 | |
|           let value = this._ini.getString(
 | |
|             "LocalizablePreferences-" + this._locale,
 | |
|             key
 | |
|           );
 | |
|           if (value) {
 | |
|             value = parseValue(value);
 | |
|             localizedStr.data = "data:text/plain," + key + "=" + value;
 | |
|             defaults._prefBranch.setComplexValue(
 | |
|               key,
 | |
|               Ci.nsIPrefLocalizedString,
 | |
|               localizedStr
 | |
|             );
 | |
|           }
 | |
|           usedLocalizablePreferences.push(key);
 | |
|         } catch (e) {
 | |
|           /* ignore bad prefs and move on */
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (sections["LocalizablePreferences-" + this._language]) {
 | |
|       for (let key of this._ini.getKeys(
 | |
|         "LocalizablePreferences-" + this._language
 | |
|       )) {
 | |
|         if (usedLocalizablePreferences.indexOf(key) > -1) {
 | |
|           continue;
 | |
|         }
 | |
|         try {
 | |
|           let value = this._ini.getString(
 | |
|             "LocalizablePreferences-" + this._language,
 | |
|             key
 | |
|           );
 | |
|           if (value) {
 | |
|             value = parseValue(value);
 | |
|             localizedStr.data = "data:text/plain," + key + "=" + value;
 | |
|             defaults._prefBranch.setComplexValue(
 | |
|               key,
 | |
|               Ci.nsIPrefLocalizedString,
 | |
|               localizedStr
 | |
|             );
 | |
|           }
 | |
|           usedLocalizablePreferences.push(key);
 | |
|         } catch (e) {
 | |
|           /* ignore bad prefs and move on */
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (sections.LocalizablePreferences) {
 | |
|       for (let key of this._ini.getKeys("LocalizablePreferences")) {
 | |
|         if (usedLocalizablePreferences.indexOf(key) > -1) {
 | |
|           continue;
 | |
|         }
 | |
|         try {
 | |
|           let value = this._ini.getString("LocalizablePreferences", key);
 | |
|           if (value) {
 | |
|             value = parseValue(value);
 | |
|             value = value.replace(/%LOCALE%/g, this._locale);
 | |
|             value = value.replace(/%LANGUAGE%/g, this._language);
 | |
|             localizedStr.data = "data:text/plain," + key + "=" + value;
 | |
|             defaults._prefBranch.setComplexValue(
 | |
|               key,
 | |
|               Ci.nsIPrefLocalizedString,
 | |
|               localizedStr
 | |
|             );
 | |
|           }
 | |
|         } catch (e) {
 | |
|           /* ignore bad prefs and move on */
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return this._checkCustomizationComplete();
 | |
|   },
 | |
| 
 | |
|   _checkCustomizationComplete: function DIST__checkCustomizationComplete() {
 | |
|     const BROWSER_DOCURL = AppConstants.BROWSER_CHROME_URL;
 | |
| 
 | |
|     if (this._newProfile) {
 | |
|       let xulStore = Services.xulStore;
 | |
| 
 | |
|       try {
 | |
|         var showPersonalToolbar = Services.prefs.getBoolPref(
 | |
|           "browser.showPersonalToolbar"
 | |
|         );
 | |
|         if (showPersonalToolbar) {
 | |
|           xulStore.setValue(
 | |
|             BROWSER_DOCURL,
 | |
|             "PersonalToolbar",
 | |
|             "collapsed",
 | |
|             "false"
 | |
|           );
 | |
|         }
 | |
|       } catch (e) {}
 | |
|       try {
 | |
|         var showMenubar = Services.prefs.getBoolPref("browser.showMenubar");
 | |
|         if (showMenubar) {
 | |
|           xulStore.setValue(
 | |
|             BROWSER_DOCURL,
 | |
|             "toolbar-menubar",
 | |
|             "autohide",
 | |
|             "false"
 | |
|           );
 | |
|         }
 | |
|       } catch (e) {}
 | |
|     }
 | |
| 
 | |
|     let prefDefaultsApplied = this._prefDefaultsApplied || !this._ini;
 | |
|     if (
 | |
|       this._customizationsApplied &&
 | |
|       this._bookmarksApplied &&
 | |
|       prefDefaultsApplied
 | |
|     ) {
 | |
|       Services.obs.notifyObservers(
 | |
|         null,
 | |
|         DISTRIBUTION_CUSTOMIZATION_COMPLETE_TOPIC
 | |
|       );
 | |
|     }
 | |
|   },
 | |
| };
 | |
| 
 | |
| function parseValue(value) {
 | |
|   try {
 | |
|     value = JSON.parse(value);
 | |
|   } catch (e) {
 | |
|     // JSON.parse catches numbers and booleans.
 | |
|     // Anything else, we assume is a string.
 | |
|     // Remove the quotes that aren't needed anymore.
 | |
|     value = value.replace(/^"/, "");
 | |
|     value = value.replace(/"$/, "");
 | |
|   }
 | |
|   return value;
 | |
| }
 | |
| 
 | |
| function enumToObject(UTF8Enumerator) {
 | |
|   let ret = {};
 | |
|   for (let i of UTF8Enumerator) {
 | |
|     ret[i] = 1;
 | |
|   }
 | |
|   return ret;
 | |
| }
 | 
