2012-08-27 19:43:57 -07:00
|
|
|
/* 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/. */
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
const Cu = Components.utils;
|
|
|
|
const Cc = Components.classes;
|
|
|
|
const Ci = Components.interfaces;
|
|
|
|
const Cr = Components.results;
|
|
|
|
|
|
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
2013-03-15 07:18:58 -07:00
|
|
|
Cu.import("resource://gre/modules/FileUtils.jsm");
|
2012-08-27 19:43:57 -07:00
|
|
|
|
2012-12-07 11:20:15 -08:00
|
|
|
XPCOMUtils.defineLazyGetter(this, "NetUtil", function() {
|
|
|
|
return Cc["@mozilla.org/network/util;1"]
|
|
|
|
.getService(Ci.nsINetUtil);
|
|
|
|
});
|
|
|
|
|
2012-09-22 17:48:26 -07:00
|
|
|
// Shared code for AppsServiceChild.jsm, Webapps.jsm and Webapps.js
|
2012-08-27 19:43:57 -07:00
|
|
|
|
2012-10-31 09:13:28 -07:00
|
|
|
this.EXPORTED_SYMBOLS = ["AppsUtils", "ManifestHelper"];
|
2012-08-27 19:43:57 -07:00
|
|
|
|
|
|
|
function debug(s) {
|
|
|
|
//dump("-*- AppsUtils.jsm: " + s + "\n");
|
|
|
|
}
|
|
|
|
|
2013-01-08 15:50:01 -08:00
|
|
|
function isAbsoluteURI(aURI) {
|
|
|
|
let foo = Services.io.newURI("http://foo", null, null);
|
|
|
|
let bar = Services.io.newURI("http://bar", null, null);
|
|
|
|
return Services.io.newURI(aURI, null, foo).prePath != foo.prePath ||
|
|
|
|
Services.io.newURI(aURI, null, bar).prePath != bar.prePath;
|
|
|
|
}
|
|
|
|
|
2013-02-06 15:21:15 -08:00
|
|
|
function mozIApplication() {
|
|
|
|
}
|
|
|
|
|
|
|
|
mozIApplication.prototype = {
|
|
|
|
hasPermission: function(aPermission) {
|
|
|
|
let uri = Services.io.newURI(this.origin, null, null);
|
|
|
|
let secMan = Cc["@mozilla.org/scriptsecuritymanager;1"]
|
|
|
|
.getService(Ci.nsIScriptSecurityManager);
|
|
|
|
// This helper checks an URI inside |aApp|'s origin and part of |aApp| has a
|
|
|
|
// specific permission. It is not checking if browsers inside |aApp| have such
|
|
|
|
// permission.
|
|
|
|
let principal = secMan.getAppCodebasePrincipal(uri, this.localId,
|
|
|
|
/*mozbrowser*/false);
|
|
|
|
let perm = Services.perms.testExactPermissionFromPrincipal(principal,
|
|
|
|
aPermission);
|
|
|
|
return (perm === Ci.nsIPermissionManager.ALLOW_ACTION);
|
|
|
|
},
|
|
|
|
|
|
|
|
QueryInterface: function(aIID) {
|
|
|
|
if (aIID.equals(Ci.mozIDOMApplication) ||
|
|
|
|
aIID.equals(Ci.mozIApplication) ||
|
|
|
|
aIID.equals(Ci.nsISupports))
|
|
|
|
return this;
|
|
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-31 09:13:28 -07:00
|
|
|
this.AppsUtils = {
|
2012-08-27 19:43:57 -07:00
|
|
|
// Clones a app, without the manifest.
|
|
|
|
cloneAppObject: function cloneAppObject(aApp) {
|
|
|
|
return {
|
2012-09-10 21:30:10 -07:00
|
|
|
name: aApp.name,
|
2012-10-19 03:43:17 -07:00
|
|
|
csp: aApp.csp,
|
2012-08-27 19:43:57 -07:00
|
|
|
installOrigin: aApp.installOrigin,
|
|
|
|
origin: aApp.origin,
|
|
|
|
receipts: aApp.receipts ? JSON.parse(JSON.stringify(aApp.receipts)) : null,
|
|
|
|
installTime: aApp.installTime,
|
|
|
|
manifestURL: aApp.manifestURL,
|
|
|
|
appStatus: aApp.appStatus,
|
2012-08-29 14:20:03 -07:00
|
|
|
removable: aApp.removable,
|
2012-12-22 05:56:21 -08:00
|
|
|
id: aApp.id,
|
2012-08-27 19:43:57 -07:00
|
|
|
localId: aApp.localId,
|
2012-12-22 05:56:21 -08:00
|
|
|
basePath: aApp.basePath,
|
2012-08-27 19:43:57 -07:00
|
|
|
progress: aApp.progress || 0.0,
|
2012-09-26 18:01:20 -07:00
|
|
|
installState: aApp.installState || "installed",
|
|
|
|
downloadAvailable: aApp.downloadAvailable,
|
|
|
|
downloading: aApp.downloading,
|
|
|
|
readyToApplyDownload: aApp.readyToApplyDownload,
|
|
|
|
downloadSize: aApp.downloadSize || 0,
|
|
|
|
lastUpdateCheck: aApp.lastUpdateCheck,
|
2012-11-16 19:37:41 -08:00
|
|
|
updateTime: aApp.updateTime,
|
2012-12-10 15:49:02 -08:00
|
|
|
etag: aApp.etag,
|
2012-12-21 10:28:58 -08:00
|
|
|
packageEtag: aApp.packageEtag,
|
2013-01-18 09:54:06 -08:00
|
|
|
manifestHash: aApp.manifestHash,
|
|
|
|
packageHash: aApp.packageHash,
|
2013-01-24 18:24:17 -08:00
|
|
|
staged: aApp.staged,
|
2012-12-10 15:49:02 -08:00
|
|
|
installerAppId: aApp.installerAppId || Ci.nsIScriptSecurityManager.NO_APP_ID,
|
2013-01-25 01:57:14 -08:00
|
|
|
installerIsBrowser: !!aApp.installerIsBrowser,
|
|
|
|
storeId: aApp.storeId || "",
|
|
|
|
storeVersion: aApp.storeVersion || 0
|
2012-08-27 19:43:57 -07:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
cloneAsMozIApplication: function cloneAsMozIApplication(aApp) {
|
|
|
|
let res = this.cloneAppObject(aApp);
|
2013-02-06 15:21:15 -08:00
|
|
|
res.__proto__ = mozIApplication.prototype;
|
2012-08-27 19:43:57 -07:00
|
|
|
return res;
|
|
|
|
},
|
|
|
|
|
|
|
|
getAppByManifestURL: function getAppByManifestURL(aApps, aManifestURL) {
|
|
|
|
debug("getAppByManifestURL " + aManifestURL);
|
|
|
|
// This could be O(1) if |webapps| was a dictionary indexed on manifestURL
|
|
|
|
// which should be the unique app identifier.
|
|
|
|
// It's currently O(n).
|
|
|
|
for (let id in aApps) {
|
|
|
|
let app = aApps[id];
|
|
|
|
if (app.manifestURL == aManifestURL) {
|
|
|
|
return this.cloneAsMozIApplication(app);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
|
|
|
getAppLocalIdByManifestURL: function getAppLocalIdByManifestURL(aApps, aManifestURL) {
|
|
|
|
debug("getAppLocalIdByManifestURL " + aManifestURL);
|
|
|
|
for (let id in aApps) {
|
|
|
|
if (aApps[id].manifestURL == aManifestURL) {
|
|
|
|
return aApps[id].localId;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ci.nsIScriptSecurityManager.NO_APP_ID;
|
|
|
|
},
|
|
|
|
|
2013-01-25 01:57:14 -08:00
|
|
|
getAppLocalIdByStoreId: function(aApps, aStoreId) {
|
|
|
|
debug("getAppLocalIdByStoreId:" + aStoreId);
|
|
|
|
for (let id in aApps) {
|
|
|
|
if (aApps[id].storeId == aStoreId) {
|
|
|
|
return aApps[id].localId;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ci.nsIScriptSecurityManager.NO_APP_ID;
|
|
|
|
},
|
|
|
|
|
2012-10-19 03:43:17 -07:00
|
|
|
getCSPByLocalId: function getCSPByLocalId(aApps, aLocalId) {
|
|
|
|
debug("getCSPByLocalId " + aLocalId);
|
|
|
|
for (let id in aApps) {
|
|
|
|
let app = aApps[id];
|
|
|
|
if (app.localId == aLocalId) {
|
|
|
|
return ( app.csp || "" );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
},
|
|
|
|
|
2012-08-27 19:43:57 -07:00
|
|
|
getAppByLocalId: function getAppByLocalId(aApps, aLocalId) {
|
|
|
|
debug("getAppByLocalId " + aLocalId);
|
|
|
|
for (let id in aApps) {
|
|
|
|
let app = aApps[id];
|
|
|
|
if (app.localId == aLocalId) {
|
|
|
|
return this.cloneAsMozIApplication(app);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
|
|
|
getManifestURLByLocalId: function getManifestURLByLocalId(aApps, aLocalId) {
|
|
|
|
debug("getManifestURLByLocalId " + aLocalId);
|
|
|
|
for (let id in aApps) {
|
|
|
|
let app = aApps[id];
|
|
|
|
if (app.localId == aLocalId) {
|
|
|
|
return app.manifestURL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
2012-08-31 07:34:28 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
getAppFromObserverMessage: function(aApps, aMessage) {
|
|
|
|
let data = JSON.parse(aMessage);
|
|
|
|
|
|
|
|
for (let id in aApps) {
|
|
|
|
let app = aApps[id];
|
|
|
|
if (app.origin != data.origin) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.cloneAsMozIApplication(app);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
2012-09-22 17:48:26 -07:00
|
|
|
},
|
|
|
|
|
2013-03-15 07:18:58 -07:00
|
|
|
getCoreAppsBasePath: function getCoreAppsBasePath() {
|
|
|
|
debug("getCoreAppsBasePath()");
|
|
|
|
try {
|
|
|
|
return FileUtils.getDir("coreAppsDir", ["webapps"], false).path;
|
|
|
|
} catch(e) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
getAppInfo: function getAppInfo(aApps, aAppId) {
|
|
|
|
if (!aApps[aAppId]) {
|
|
|
|
debug("No webapp for " + aAppId);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can have 3rd party apps that are non-removable,
|
|
|
|
// so we can't use the 'removable' property for isCoreApp
|
|
|
|
// Instead, we check if the app is installed under /system/b2g
|
|
|
|
let isCoreApp = false;
|
|
|
|
let app = aApps[aAppId];
|
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
isCoreApp = app.basePath == this.getCoreAppsBasePath();
|
|
|
|
#endif
|
|
|
|
debug(app.name + " isCoreApp: " + isCoreApp);
|
|
|
|
return { "basePath": app.basePath + "/",
|
|
|
|
"isCoreApp": isCoreApp };
|
|
|
|
},
|
|
|
|
|
2013-04-18 03:18:30 -07:00
|
|
|
/**
|
|
|
|
* Remove potential HTML tags from displayable fields in the manifest.
|
|
|
|
* We check name, description, developer name, and permission description
|
|
|
|
*/
|
|
|
|
sanitizeManifest: function(aManifest) {
|
|
|
|
let sanitizer = Cc["@mozilla.org/parserutils;1"]
|
|
|
|
.getService(Ci.nsIParserUtils);
|
|
|
|
if (!sanitizer) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
function sanitize(aStr) {
|
|
|
|
return sanitizer.convertToPlainText(aStr,
|
|
|
|
Ci.nsIDocumentEncoder.OutputRaw, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
function sanitizeEntryPoint(aRoot) {
|
|
|
|
aRoot.name = sanitize(aRoot.name);
|
|
|
|
|
|
|
|
if (aRoot.description) {
|
|
|
|
aRoot.description = sanitize(aRoot.description);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aRoot.developer && aRoot.developer.name) {
|
|
|
|
aRoot.developer.name = sanitize(aRoot.developer.name);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aRoot.permissions) {
|
|
|
|
for (let permission in aRoot.permissions) {
|
|
|
|
if (aRoot.permissions[permission].description) {
|
|
|
|
aRoot.permissions[permission].description =
|
|
|
|
sanitize(aRoot.permissions[permission].description);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// First process the main section, then the entry points.
|
|
|
|
sanitizeEntryPoint(aManifest);
|
|
|
|
|
|
|
|
if (aManifest.entry_points) {
|
|
|
|
for (let entry in aManifest.entry_points) {
|
|
|
|
sanitizeEntryPoint(aManifest.entry_points[entry]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-09-22 17:48:26 -07:00
|
|
|
/**
|
2013-02-15 10:13:02 -08:00
|
|
|
* From https://developer.mozilla.org/en/OpenWebApps/The_Manifest
|
|
|
|
* Only the name property is mandatory.
|
2012-09-22 17:48:26 -07:00
|
|
|
*/
|
2013-01-11 06:08:12 -08:00
|
|
|
checkManifest: function(aManifest, app) {
|
2012-09-22 17:48:26 -07:00
|
|
|
if (aManifest.name == undefined)
|
|
|
|
return false;
|
|
|
|
|
2013-04-18 03:18:30 -07:00
|
|
|
this.sanitizeManifest(aManifest);
|
|
|
|
|
2013-01-08 15:50:01 -08:00
|
|
|
// launch_path, entry_points launch paths, message hrefs, and activity hrefs can't be absolute
|
|
|
|
if (aManifest.launch_path && isAbsoluteURI(aManifest.launch_path))
|
2012-09-22 17:48:26 -07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
function checkAbsoluteEntryPoints(entryPoints) {
|
|
|
|
for (let name in entryPoints) {
|
2013-01-08 15:50:01 -08:00
|
|
|
if (entryPoints[name].launch_path && isAbsoluteURI(entryPoints[name].launch_path)) {
|
2012-09-22 17:48:26 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (checkAbsoluteEntryPoints(aManifest.entry_points))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (let localeName in aManifest.locales) {
|
|
|
|
if (checkAbsoluteEntryPoints(aManifest.locales[localeName].entry_points)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-08 15:50:01 -08:00
|
|
|
if (aManifest.activities) {
|
|
|
|
for (let activityName in aManifest.activities) {
|
|
|
|
let activity = aManifest.activities[activityName];
|
|
|
|
if (activity.href && isAbsoluteURI(activity.href)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// |messages| is an array of items, where each item is either a string or
|
|
|
|
// a {name: href} object.
|
|
|
|
let messages = aManifest.messages;
|
|
|
|
if (messages) {
|
|
|
|
if (!Array.isArray(messages)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (let item of aManifest.messages) {
|
|
|
|
if (typeof item == "object") {
|
|
|
|
let keys = Object.keys(item);
|
|
|
|
if (keys.length != 1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (isAbsoluteURI(item[keys[0]])) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-15 10:13:02 -08:00
|
|
|
// The 'size' field must be a positive integer.
|
|
|
|
if (aManifest.size) {
|
|
|
|
aManifest.size = parseInt(aManifest.size);
|
|
|
|
if (Number.isNaN(aManifest.size) || aManifest.size < 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-22 17:48:26 -07:00
|
|
|
return true;
|
2012-10-02 22:38:06 -07:00
|
|
|
},
|
|
|
|
|
2012-11-29 14:52:02 -08:00
|
|
|
checkManifestContentType: function
|
2012-12-07 11:20:15 -08:00
|
|
|
checkManifestContentType(aInstallOrigin, aWebappOrigin, aContentType) {
|
|
|
|
let hadCharset = { };
|
|
|
|
let charset = { };
|
|
|
|
let contentType = NetUtil.parseContentType(aContentType, charset, hadCharset);
|
|
|
|
if (aInstallOrigin != aWebappOrigin &&
|
2012-11-29 14:52:02 -08:00
|
|
|
contentType != "application/x-web-app-manifest+json") {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
2013-01-11 06:08:12 -08:00
|
|
|
/**
|
|
|
|
* Method to apply modifications to webapp manifests file saved internally.
|
|
|
|
* For now, only ensure app can't rename itself.
|
|
|
|
*/
|
2013-02-14 09:23:58 -08:00
|
|
|
ensureSameAppName: function ensureSameAppName(aOldManifest, aNewManifest, aApp) {
|
2013-01-11 06:08:12 -08:00
|
|
|
// Ensure that app name can't be updated
|
2013-02-14 09:23:58 -08:00
|
|
|
aNewManifest.name = aApp.name;
|
2013-01-11 06:08:12 -08:00
|
|
|
|
|
|
|
// Nor through localized names
|
2013-02-14 09:23:58 -08:00
|
|
|
if ('locales' in aNewManifest) {
|
|
|
|
let defaultName = new ManifestHelper(aOldManifest, aApp.origin).name;
|
|
|
|
for (let locale in aNewManifest.locales) {
|
|
|
|
let entry = aNewManifest.locales[locale];
|
2013-01-11 06:08:12 -08:00
|
|
|
if (!entry.name) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// In case previous manifest didn't had a name,
|
|
|
|
// we use the default app name
|
|
|
|
let localizedName = defaultName;
|
2013-02-14 09:23:58 -08:00
|
|
|
if (aOldManifest && 'locales' in aOldManifest &&
|
|
|
|
locale in aOldManifest.locales) {
|
|
|
|
localizedName = aOldManifest.locales[locale].name;
|
2013-01-11 06:08:12 -08:00
|
|
|
}
|
|
|
|
entry.name = localizedName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-10-02 22:38:06 -07:00
|
|
|
/**
|
2012-11-18 14:43:38 -08:00
|
|
|
* Determines whether the manifest allows installs for the given origin.
|
|
|
|
* @param object aManifest
|
|
|
|
* @param string aInstallOrigin
|
|
|
|
* @return boolean
|
|
|
|
**/
|
|
|
|
checkInstallAllowed: function checkInstallAllowed(aManifest, aInstallOrigin) {
|
|
|
|
if (!aManifest.installs_allowed_from) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
function cbCheckAllowedOrigin(aOrigin) {
|
|
|
|
return aOrigin == "*" || aOrigin == aInstallOrigin;
|
|
|
|
}
|
|
|
|
|
|
|
|
return aManifest.installs_allowed_from.some(cbCheckAllowedOrigin);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine the type of app (app, privileged, certified)
|
|
|
|
* that is installed by the manifest
|
|
|
|
* @param object aManifest
|
|
|
|
* @returns integer
|
|
|
|
**/
|
2012-10-02 22:38:06 -07:00
|
|
|
getAppManifestStatus: function getAppManifestStatus(aManifest) {
|
|
|
|
let type = aManifest.type || "web";
|
|
|
|
|
|
|
|
switch(type) {
|
|
|
|
case "web":
|
|
|
|
return Ci.nsIPrincipal.APP_STATUS_INSTALLED;
|
|
|
|
case "privileged":
|
|
|
|
return Ci.nsIPrincipal.APP_STATUS_PRIVILEGED;
|
|
|
|
case "certified":
|
|
|
|
return Ci.nsIPrincipal.APP_STATUS_CERTIFIED;
|
|
|
|
default:
|
|
|
|
throw new Error("Webapps.jsm: Undetermined app manifest type");
|
|
|
|
}
|
2012-10-10 09:16:49 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determines if an update or a factory reset occured.
|
|
|
|
*/
|
|
|
|
isFirstRun: function isFirstRun(aPrefBranch) {
|
|
|
|
let savedmstone = null;
|
|
|
|
try {
|
|
|
|
savedmstone = aPrefBranch.getCharPref("gecko.mstone");
|
|
|
|
} catch (e) {}
|
|
|
|
|
|
|
|
let mstone = Services.appinfo.platformVersion;
|
|
|
|
|
|
|
|
let savedBuildID = null;
|
|
|
|
try {
|
|
|
|
savedBuildID = aPrefBranch.getCharPref("gecko.buildID");
|
|
|
|
} catch (e) {}
|
|
|
|
|
|
|
|
let buildID = Services.appinfo.platformBuildID;
|
|
|
|
|
|
|
|
aPrefBranch.setCharPref("gecko.mstone", mstone);
|
|
|
|
aPrefBranch.setCharPref("gecko.buildID", buildID);
|
|
|
|
|
|
|
|
return ((mstone != savedmstone) || (buildID != savedBuildID));
|
|
|
|
},
|
2012-12-19 11:37:13 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if two manifests have the same set of properties and that the
|
|
|
|
* values of these properties are the same, in each locale.
|
|
|
|
* Manifests here are raw json ones.
|
|
|
|
*/
|
|
|
|
compareManifests: function compareManifests(aManifest1, aManifest2) {
|
|
|
|
// 1. check if we have the same locales in both manifests.
|
|
|
|
let locales1 = [];
|
|
|
|
let locales2 = [];
|
|
|
|
if (aManifest1.locales) {
|
|
|
|
for (let locale in aManifest1.locales) {
|
|
|
|
locales1.push(locale);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (aManifest2.locales) {
|
|
|
|
for (let locale in aManifest2.locales) {
|
|
|
|
locales2.push(locale);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (locales1.sort().join() !== locales2.sort().join()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper function to check the app name and developer information for
|
|
|
|
// two given roots.
|
|
|
|
let checkNameAndDev = function(aRoot1, aRoot2) {
|
|
|
|
let name1 = aRoot1.name;
|
|
|
|
let name2 = aRoot2.name;
|
|
|
|
if (name1 !== name2) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
let dev1 = aRoot1.developer;
|
|
|
|
let dev2 = aRoot2.developer;
|
|
|
|
if ((dev1 && !dev2) || (dev2 && !dev1)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-01-10 21:47:00 -08:00
|
|
|
return (!dev1 && !dev2) ||
|
|
|
|
(dev1.name === dev2.name && dev1.url === dev2.url);
|
2012-12-19 11:37:13 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// 2. For each locale, check if the name and dev info are the same.
|
|
|
|
if (!checkNameAndDev(aManifest1, aManifest2)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let locale in aManifest1.locales) {
|
|
|
|
if (!checkNameAndDev(aManifest1.locales[locale],
|
|
|
|
aManifest2.locales[locale])) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Nothing failed.
|
|
|
|
return true;
|
|
|
|
}
|
2012-08-27 19:43:57 -07:00
|
|
|
}
|
2012-10-02 22:38:03 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper object to access manifest information with locale support
|
|
|
|
*/
|
2012-10-31 09:13:28 -07:00
|
|
|
this.ManifestHelper = function(aManifest, aOrigin) {
|
2012-10-02 22:38:03 -07:00
|
|
|
this._origin = Services.io.newURI(aOrigin, null, null);
|
|
|
|
this._manifest = aManifest;
|
|
|
|
let chrome = Cc["@mozilla.org/chrome/chrome-registry;1"].getService(Ci.nsIXULChromeRegistry)
|
|
|
|
.QueryInterface(Ci.nsIToolkitChromeRegistry);
|
2012-11-15 02:30:48 -08:00
|
|
|
let locale = chrome.getSelectedLocale("global").toLowerCase();
|
2012-10-02 22:38:03 -07:00
|
|
|
this._localeRoot = this._manifest;
|
|
|
|
|
|
|
|
if (this._manifest.locales && this._manifest.locales[locale]) {
|
|
|
|
this._localeRoot = this._manifest.locales[locale];
|
|
|
|
}
|
|
|
|
else if (this._manifest.locales) {
|
|
|
|
// try with the language part of the locale ("en" for en-GB) only
|
|
|
|
let lang = locale.split('-')[0];
|
|
|
|
if (lang != locale && this._manifest.locales[lang])
|
|
|
|
this._localeRoot = this._manifest.locales[lang];
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
ManifestHelper.prototype = {
|
|
|
|
_localeProp: function(aProp) {
|
|
|
|
if (this._localeRoot[aProp] != undefined)
|
|
|
|
return this._localeRoot[aProp];
|
|
|
|
return this._manifest[aProp];
|
|
|
|
},
|
|
|
|
|
|
|
|
get name() {
|
|
|
|
return this._localeProp("name");
|
|
|
|
},
|
|
|
|
|
|
|
|
get description() {
|
|
|
|
return this._localeProp("description");
|
|
|
|
},
|
|
|
|
|
|
|
|
get version() {
|
|
|
|
return this._localeProp("version");
|
|
|
|
},
|
|
|
|
|
|
|
|
get launch_path() {
|
|
|
|
return this._localeProp("launch_path");
|
|
|
|
},
|
|
|
|
|
|
|
|
get developer() {
|
2013-01-10 21:47:00 -08:00
|
|
|
// Default to {} in order to avoid exception in code
|
|
|
|
// that doesn't check for null `developer`
|
|
|
|
return this._localeProp("developer") || {};
|
2012-10-02 22:38:03 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
get icons() {
|
|
|
|
return this._localeProp("icons");
|
|
|
|
},
|
|
|
|
|
|
|
|
get appcache_path() {
|
|
|
|
return this._localeProp("appcache_path");
|
|
|
|
},
|
|
|
|
|
|
|
|
get orientation() {
|
|
|
|
return this._localeProp("orientation");
|
|
|
|
},
|
|
|
|
|
|
|
|
get package_path() {
|
|
|
|
return this._localeProp("package_path");
|
|
|
|
},
|
|
|
|
|
|
|
|
get size() {
|
|
|
|
return this._manifest["size"] || 0;
|
|
|
|
},
|
|
|
|
|
|
|
|
get permissions() {
|
|
|
|
if (this._manifest.permissions) {
|
|
|
|
return this._manifest.permissions;
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
},
|
|
|
|
|
|
|
|
iconURLForSize: function(aSize) {
|
|
|
|
let icons = this._localeProp("icons");
|
|
|
|
if (!icons)
|
|
|
|
return null;
|
|
|
|
let dist = 100000;
|
|
|
|
let icon = null;
|
|
|
|
for (let size in icons) {
|
|
|
|
let iSize = parseInt(size);
|
|
|
|
if (Math.abs(iSize - aSize) < dist) {
|
|
|
|
icon = this._origin.resolve(icons[size]);
|
|
|
|
dist = Math.abs(iSize - aSize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return icon;
|
|
|
|
},
|
|
|
|
|
|
|
|
fullLaunchPath: function(aStartPoint) {
|
|
|
|
// If no start point is specified, we use the root launch path.
|
|
|
|
// In all error cases, we just return null.
|
|
|
|
if ((aStartPoint || "") === "") {
|
|
|
|
return this._origin.resolve(this._localeProp("launch_path") || "");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search for the l10n entry_points property.
|
|
|
|
let entryPoints = this._localeProp("entry_points");
|
|
|
|
if (!entryPoints) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entryPoints[aStartPoint]) {
|
|
|
|
return this._origin.resolve(entryPoints[aStartPoint].launch_path || "");
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
|
|
|
resolveFromOrigin: function(aURI) {
|
2013-01-08 15:50:01 -08:00
|
|
|
// This should be enforced higher up, but check it here just in case.
|
2013-01-08 23:35:52 -08:00
|
|
|
if (isAbsoluteURI(aURI)) {
|
2013-01-08 15:50:01 -08:00
|
|
|
throw new Error("Webapps.jsm: non-relative URI passed to resolveFromOrigin");
|
|
|
|
}
|
2012-10-02 22:38:03 -07:00
|
|
|
return this._origin.resolve(aURI);
|
|
|
|
},
|
|
|
|
|
|
|
|
fullAppcachePath: function() {
|
|
|
|
let appcachePath = this._localeProp("appcache_path");
|
|
|
|
return this._origin.resolve(appcachePath ? appcachePath : "");
|
|
|
|
},
|
|
|
|
|
|
|
|
fullPackagePath: function() {
|
|
|
|
let packagePath = this._localeProp("package_path");
|
|
|
|
return this._origin.resolve(packagePath ? packagePath : "");
|
|
|
|
}
|
|
|
|
}
|