gecko-dev/toolkit/components/cloudstorage/CloudStorage.jsm

698 строки
23 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/. */
/**
* Java Script module that helps consumers store data directly
* to cloud storage provider download folders.
*
* Takes cloud storage providers metadata as JSON input on Mac, Linux and Windows.
*
* Handles scan, prompt response save and exposes preferred storage provider.
*/
"use strict";
var EXPORTED_SYMBOLS = ["CloudStorage"];
const {AppConstants} = ChromeUtils.import("resource://gre/modules/AppConstants.jsm");
const {Services} = ChromeUtils.import("resource://gre/modules/Services.jsm");
const {XPCOMUtils} = ChromeUtils.import("resource://gre/modules/XPCOMUtils.jsm");
XPCOMUtils.defineLazyGlobalGetters(this, ["fetch"]);
ChromeUtils.defineModuleGetter(this, "Downloads",
"resource://gre/modules/Downloads.jsm");
ChromeUtils.defineModuleGetter(this, "FileUtils",
"resource://gre/modules/FileUtils.jsm");
ChromeUtils.defineModuleGetter(this, "OS",
"resource://gre/modules/osfile.jsm");
const CLOUD_SERVICES_PREF = "cloud.services.";
const CLOUD_PROVIDERS_URI = "resource://cloudstorage/providers.json";
/**
* Provider metadata JSON is loaded from resource://cloudstorage/providers.json
* Sample providers.json format
*
* {
* "Dropbox": {
* "displayName": "Dropbox",
* "relativeDownloadPath": ["homeDir", "Dropbox"],
* "relativeDiscoveryPath": {
* "linux": ["homeDir", ".dropbox", "info.json"],
* "macosx": ["homeDir", ".dropbox", "info.json"],
* "win": ["LocalAppData", "Dropbox", "info.json"]
* },
* "typeSpecificData": {
* "default": "Downloads",
* "screenshot": "Screenshots"
* }
* }
*
* Providers JSON is flat list of providers metdata with property as key in format @Provider
*
* @Provider - Unique cloud provider key, possible values: "Dropbox", "GDrive"
*
* @displayName - cloud storage name displayed in the prompt.
*
* @relativeDownloadPath - download path on user desktop for a cloud storage provider.
* By default downloadPath is a concatenation of home dir and name of dropbox folder.
* Example value: ["homeDir", "Dropbox"]
*
* @relativeDiscoveryPath - Lists discoveryPath by platform. Provider is not supported on a platform
* if its value doesn't exist in relativeDiscoveryPath. relativeDiscoveryPath by platform is stored
* as an array ofsubdirectories, which when concatenated, forms discovery path.
* During scan discoveryPath is checked for the existence of cloud storage provider on user desktop.
*
* @typeSpecificData - provides folder name for a cloud storage depending
* on type of data downloaded. Default folder is 'Downloads'. Other options are
* 'screenshot' depending on provider support.
*/
/**
*
* Internal cloud services prefs
*
* cloud.services.api.enabled - set to true to initialize and use Cloud Storage module
*
* cloud.services.storage.key - set to string with preferred provider key
*
* cloud.services.lastPrompt - set to time when last prompt was shown
*
* cloud.services.interval.prompt - set to time interval in days after which prompt should be shown
*
* cloud.services.rejected.key - set to string with comma separated provider keys rejected
* by user when prompted to opt-in
*
* browser.download.folderList - set to int and indicates the location users wish to save downloaded files to.
* 0 - The desktop is the default download location.
* 1 - The system's downloads folder is the default download location.
* 2 - The default download location is elsewhere as specified in
* browser.download.dir.
* 3 - The default download location is elsewhere as specified by
* cloud storage API getDownloadFolder
*
* browser.download.dir - local file handle
* A local folder user may have selected for downloaded files to be
* saved. This folder is enabled when folderList equals 2.
*/
/**
* The external API exported by this module.
*/
var CloudStorage = {
/**
* Init method to initialize providers metadata
*/
async init() {
let isInitialized = null;
try {
// Invoke internal method asynchronously to read and
// parse providers metadata from JSON
isInitialized = await CloudStorageInternal.initProviders();
} catch (err) {
Cu.reportError(err);
}
return isInitialized;
},
/**
* Returns information to allow the consumer to decide whether showing
* a doorhanger prompt is appropriate. If a preferred provider is set
* on desktop, user is not prompted again and method returns null.
*
* @return {Promise} which resolves to an object with property name
* as 'key' and 'value'.
* 'key' property is provider key such as 'Dropbox', 'GDrive'.
* 'value' property contains metadata for respective provider.
* Resolves null if it's not appropriate to prompt.
*/
promisePromptInfo() {
return CloudStorageInternal.promisePromptInfo();
},
/**
* Save user response from doorhanger prompt.
* If user confirms and checks 'always remember', update prefs
* cloud.services.storage.key and browser.download.folderList to pick
* download location from cloud storage API
* If user denies, save provider as rejected in cloud.services.rejected.key
*
* @param key
* cloud storage provider key from provider metadata
* @param remember
* bool value indicating whether user has asked to always remember
* the settings
* @param selected
* bool value by default set to false indicating if user has selected
* to save downloaded file with cloud provider
*/
savePromptResponse(key, remember, selected = false) {
Services.prefs.setIntPref(CLOUD_SERVICES_PREF + "lastprompt",
Math.floor(Date.now() / 1000));
if (remember) {
if (selected) {
CloudStorageInternal.setCloudStoragePref(key);
} else {
// Store provider as rejected by setting cloud.services.rejected.key
// and not use in re-prompt
CloudStorageInternal.handleRejected(key);
}
}
},
/**
* Retrieve download folder of an opted-in storage provider
* by type specific data
* @param typeSpecificData
* type of data downloaded, options are 'default', 'screenshot'
* @return {Promise} which resolves to full path to provider download folder
*/
getDownloadFolder(typeSpecificData) {
return CloudStorageInternal.getDownloadFolder(typeSpecificData);
},
/**
* Get key of provider opted-in by user to store downloaded files
*
* @return {String}
* Storage provider key from provider metadata. Return empty string
* if user has not selected a preferred provider.
*/
getPreferredProvider() {
return CloudStorageInternal.preferredProviderKey;
},
/**
* Get metadata of provider opted-in by user to store downloaded files.
* Return preferred provider metadata without scanning by doing simple lookup
* inside storage providers metadata using preferred provider key
*
* @return {Object}
* Object with preferred provider metadata. Return null
* if user has not selected a preferred provider.
*/
getPreferredProviderMetaData() {
return CloudStorageInternal.getPreferredProviderMetaData();
},
/**
* Get display name of a provider actively in use to store downloaded files
*
* @return {String}
* String with provider display name. Returns null if a provider
* is not in use.
*/
getProviderIfInUse() {
return CloudStorageInternal.getProviderIfInUse();
},
/**
* Get providers found on user desktop. Used for unit tests
*
* @return {Promise}
* @resolves
* Map object with entries key set to storage provider key and values set to
* storage provider metadata
*/
getStorageProviders() {
return CloudStorageInternal.getStorageProviders();
},
};
/**
* The internal API for the CloudStorage module.
*/
var CloudStorageInternal = {
/**
* promiseInit saves returned init method promise and is
* used to wait for initialization to complete.
*/
promiseInit: null,
/**
* Internal property having storage providers data
*/
providersMetaData: null,
async _downloadJSON(uri) {
let json = null;
try {
let response = await fetch(uri);
if (response.ok) {
json = await response.json();
}
} catch (e) {
Cu.reportError("Fetching " + uri + " results in error: " + e);
}
return json;
},
/**
* Reset 'browser.download.folderList' cloud storage value '3' back
* to '2' or '1' depending on custom path or system default Downloads path
* in pref 'browser.download.dir'.
*/
async resetFolderListPref() {
let folderListValue = Services.prefs.getIntPref("browser.download.folderList", 0);
if (folderListValue !== 3) {
return;
}
let downloadDirPath = null;
try {
let file = Services.prefs.getComplexValue("browser.download.dir",
Ci.nsIFile);
downloadDirPath = file.path;
} catch (e) {}
if (!downloadDirPath ||
(downloadDirPath === await Downloads.getSystemDownloadsDirectory())) {
// if downloadDirPath is the Downloads folder path or unspecified
folderListValue = 1;
} else if (downloadDirPath === Services.dirsvc.get("Desk", Ci.nsIFile).path) {
// if downloadDirPath is the Desktop path
folderListValue = 0;
} else {
// otherwise
folderListValue = 2;
}
Services.prefs.setIntPref("browser.download.folderList", folderListValue);
},
/**
* Loads storage providers metadata asynchronously from providers.json.
*
* @returns {Promise} with resolved boolean value true if providers
* metadata is successfully initialized
*/
async initProviders() {
// Cloud Storage API should continue initialization and load providers metadata
// only if a consumer add-on using API sets pref 'cloud.services.api.enabled' to true
// If API is not enabled, check and reset cloud storage value in folderList pref.
if (!this.isAPIEnabled) {
this.resetFolderListPref().catch((err) => {
Cu.reportError("CloudStorage: Failed to reset folderList pref " + err);
});
return false;
}
let response = await this._downloadJSON(CLOUD_PROVIDERS_URI);
this.providersMetaData = await this._parseProvidersJSON(response);
let providersCount = Object.keys(this.providersMetaData).length;
if (providersCount > 0) {
// Array of boolean results for each provider handled for custom downloadpath
let handledProviders = await this.initDownloadPathIfProvidersExist();
if (handledProviders.length === providersCount) {
return true;
}
}
return false;
},
/**
* Load parsed metadata inside providers object
*/
_parseProvidersJSON(providers) {
if (!providers) {
return {};
}
// Use relativeDiscoveryPath to filter providers object by platform.
// DownloadPath and discoveryPath are stored as
// array of subdirectories inside providers.json
// Update providers object discoveryPath and downloadPath
// property values by concatenating subdirectories and forming platform
// specific directory path
Object.getOwnPropertyNames(providers).forEach(key => {
if (providers[key].relativeDiscoveryPath.hasOwnProperty(AppConstants.platform)) {
providers[key].discoveryPath =
this._concatPath(providers[key].relativeDiscoveryPath[AppConstants.platform]);
providers[key].downloadPath =
this._concatPath(providers[key].relativeDownloadPath);
} else {
// delete key not supported on AppConstants.platform
delete providers[key];
}
});
return providers;
},
/**
* Concatenate subdir value inside array to form
* platform specific directory path
*
* @param arrDirs
* String Array containing sub directories name
* @returns Path of type String
*/
_concatPath(arrDirs) {
let dirPath = "";
for (let subDir of arrDirs) {
switch (subDir) {
case "homeDir":
subDir = OS.Constants.Path.homeDir ? OS.Constants.Path.homeDir : "";
break;
case "LocalAppData":
if (OS.Constants.Win) {
let nsIFileLocal = Services.dirsvc.get("LocalAppData", Ci.nsIFile);
subDir = nsIFileLocal && nsIFileLocal.path ? nsIFileLocal.path : "";
} else {
subDir = "";
}
break;
}
dirPath = OS.Path.join(dirPath, subDir);
}
return dirPath;
},
/**
* Check for custom download paths and override providers metadata
* downloadPath property
*
* For dropbox open config file ~/.dropbox/info.json
* and override downloadPath with path found
* See https://www.dropbox.com/en/help/desktop-web/find-folder-paths
*
* For all other providers we are using downloadpath from providers.json
*
* @returns {Promise} with array boolean values for respective provider. Value is true if a
* provider exist on user desktop and its downloadPath is updated. Promise returns with
* resolved array value when all providers in metadata are handled.
*/
initDownloadPathIfProvidersExist() {
let providerKeys = Object.keys(this.providersMetaData);
let promises = providerKeys.map(key => {
return key === "Dropbox" ?
this._initDropbox(key) :
Promise.resolve(false);
});
return Promise.all(promises);
},
/**
* Read Dropbox info.json and override providers metadata
* downloadPath property
*
* @return {Promise}
* @resolves
* false if dropbox provider is not found. Returns true if dropbox service exist
* on user desktop and downloadPath in providermetadata is updated with
* value read from config file info.json
*/
async _initDropbox(key) {
// Check if Dropbox provider exist on desktop before continuing
if (!await this._checkIfAssetExists(this.providersMetaData[key].discoveryPath)) {
return false;
}
// Check in cloud.services.rejected.key if Dropbox is previously rejected before continuing
let rejectedKeys = this.cloudStorageRejectedKeys.split(",");
if (rejectedKeys.includes(key)) {
return false;
}
let file = null;
try {
file = new FileUtils.File(this.providersMetaData[key].discoveryPath);
} catch (ex) {
return false;
}
let data = await this._downloadJSON(Services.io.newFileURI(file).spec);
if (!data) {
return false;
}
let path = data && data.personal && data.personal.path;
if (!path) {
return false;
}
let isUsable = await this._isUsableDirectory(path);
if (isUsable) {
this.providersMetaData.Dropbox.downloadPath = path;
}
return isUsable;
},
/**
* Determines if a given directory is valid and can be used to download files
*
* @param full absolute path to the directory
*
* @return {Promise} which resolves true if we can use the directory, false otherwise.
*/
async _isUsableDirectory(path) {
let isUsable = false;
try {
let info = await OS.File.stat(path);
isUsable = info.isDir;
} catch (e) {
// Directory doesn't exist, so isUsable will still be false
}
return isUsable;
},
/**
* Retrieve download folder of preferred provider by type specific data
*
* @param dataType
* type of data downloaded, options are 'default', 'screenshot'
* default value is 'default'
* @return {Promise} which resolves to full path to download folder
* Resolves null if a valid download folder is not found.
*/
async getDownloadFolder(dataType = "default") {
// Wait for cloudstorage to initialize if providers metadata is not available
if (!this.providersMetaData) {
let isInitialized = await this.promiseInit;
if (!isInitialized && !this.providersMetaData) {
Cu.reportError("CloudStorage: Failed to initialize and retrieve download folder ");
return null;
}
}
let key = this.preferredProviderKey;
if (!key || !this.providersMetaData.hasOwnProperty(key)) {
return null;
}
let provider = this.providersMetaData[key];
if (!provider.typeSpecificData[dataType]) {
return null;
}
let downloadDirPath = OS.Path.join(provider.downloadPath,
provider.typeSpecificData[dataType]);
if (!(await this._isUsableDirectory(downloadDirPath))) {
return null;
}
return downloadDirPath;
},
/**
* Return scanned provider info used by consumer inside doorhanger prompt.
* @return {Promise}
* which resolves to an object with property 'key' as found provider and
* property 'value' as provider metadata.
* Resolves null if no provider info is returned.
*/
async promisePromptInfo() {
// Check if user has not previously opted-in for preferred provider download folder
// and if time elapsed since last prompt shown has exceeded maximum allowed interval
// in pref cloud.services.interval.prompt before continuing to scan for providers
if (!this.preferredProviderKey && this.shouldPrompt()) {
return this.scan();
}
return Promise.resolve(null);
},
/**
* Check if its time to prompt by reading lastprompt service pref.
* Return true if pref doesn't exist or last prompt time is
* more than prompt interval
*/
shouldPrompt() {
let lastPrompt = this.lastPromptTime;
let now = Math.floor(Date.now() / 1000);
let interval = now - lastPrompt;
// Convert prompt interval to seconds
let maxAllow = this.promptInterval * 24 * 60 * 60;
return interval >= maxAllow;
},
/**
* Scans for local storage providers available on user desktop
*
* providers list is read in order as specified in providers.json.
* If a user has multiple cloud storage providers on desktop, return the first
* provider after filtering the rejected keys
*
* @return {Promise}
* which resolves to an object providerInfo with found provider key and value
* as provider metadata. Resolves null if no valid provider found
*/
async scan() {
let providers = await this.getStorageProviders();
if (!providers.size) {
// No storage services installed on user desktop
return null;
}
// Filter the rejected providers in cloud.services.rejected.key
// from the providers map object
let rejectedKeys = this.cloudStorageRejectedKeys.split(",");
for (let rejectedKey of rejectedKeys) {
providers.delete(rejectedKey);
}
// Pick first storage provider from providers
let provider = providers.entries().next().value;
if (provider) {
return {key: provider[0], value: provider[1]};
}
return null;
},
/**
* Checks if the asset with input path exist on
* file system
* @return {Promise}
* @resolves
* boolean value of file existence check
*/
_checkIfAssetExists(path) {
return OS.File.exists(path).catch(err => {
Cu.reportError(`Couldn't check existance of ${path}`, err);
return false;
});
},
/**
* get access to all local storage providers available on user desktop
*
* @return {Promise}
* @resolves
* Map object with entries key set to storage provider key and values set to
* storage provider metadata
*/
async getStorageProviders() {
let providers = Object.entries(this.providersMetaData || {});
// Array of promises with boolean value exist for respective storage.
let promises = providers.map(([, provider]) => this._checkIfAssetExists(provider.discoveryPath));
let results = await Promise.all(promises);
// Filter providers array to remove provider with discoveryPath asset exist resolved value false
providers = providers.filter((_, idx) => results[idx]);
return new Map(providers);
},
/**
* Save the rejected provider in cloud.services.rejected.key. Pref
* stores rejected keys value as comma separated string.
*
* @param key
* Provider key to be saved in cloud.services.rejected.key pref
*/
handleRejected(key) {
let rejected = this.cloudStorageRejectedKeys;
if (!rejected) {
Services.prefs.setCharPref(CLOUD_SERVICES_PREF + "rejected.key", key);
} else {
// Pref exists with previous rejected keys, append
// key at the end and update pref
let keys = rejected.split(",");
if (key) {
keys.push(key);
}
Services.prefs.setCharPref(CLOUD_SERVICES_PREF + "rejected.key", keys.join(","));
}
},
/**
*
* Sets pref cloud.services.storage.key. It updates download browser.download.folderList
* value to 3 indicating download location is stored elsewhere, as specified by
* cloud storage API getDownloadFolder
*
* @param key
* cloud storage provider key from provider metadata
*/
setCloudStoragePref(key) {
Services.prefs.setCharPref(CLOUD_SERVICES_PREF + "storage.key", key);
Services.prefs.setIntPref("browser.download.folderList", 3);
},
/**
* get access to preferred provider metadata by using preferred provider key
*
* @return {Object}
* Object with preferred provider metadata. Returns null if preferred provider is not set
*/
getPreferredProviderMetaData() {
// Use preferred provider key to retrieve metadata from ProvidersMetaData
return this.providersMetaData.hasOwnProperty(this.preferredProviderKey) ?
this.providersMetaData[this.preferredProviderKey] : null;
},
/**
* Get provider display name if cloud storage API is used by an add-on
* and user has set preferred provider and a valid download directory
* path exists on user desktop.
*
* @return {String}
* String with preferred provider display name. Returns null if provider is not in use.
*/
async getProviderIfInUse() {
// Check if consumer add-on is present and user has set preferred provider key
// and a valid download path exist on user desktop
if (this.isAPIEnabled && this.preferredProviderKey && await this.getDownloadFolder()) {
let provider = this.getPreferredProviderMetaData();
return provider.displayName || null;
}
return null;
},
};
/**
* Provider key retrieved from service pref cloud.services.storage.key
*/
XPCOMUtils.defineLazyPreferenceGetter(CloudStorageInternal, "preferredProviderKey",
CLOUD_SERVICES_PREF + "storage.key", "");
/**
* Provider keys rejected by user for default download
*/
XPCOMUtils.defineLazyPreferenceGetter(CloudStorageInternal, "cloudStorageRejectedKeys",
CLOUD_SERVICES_PREF + "rejected.key", "");
/**
* Lastprompt time in seconds, by default set to 0
*/
XPCOMUtils.defineLazyPreferenceGetter(CloudStorageInternal, "lastPromptTime",
CLOUD_SERVICES_PREF + "lastprompt", 0 /* 0 second */);
/**
* show prompt interval in days, by default set to 0
*/
XPCOMUtils.defineLazyPreferenceGetter(CloudStorageInternal, "promptInterval",
CLOUD_SERVICES_PREF + "interval.prompt", 0 /* 0 days */);
/**
* generic pref that shows if cloud storage API is in use, by default set to false.
* Re-run CloudStorage init evertytime pref is set.
*/
XPCOMUtils.defineLazyPreferenceGetter(CloudStorageInternal, "isAPIEnabled",
CLOUD_SERVICES_PREF + "api.enabled", false, () => CloudStorage.init());
CloudStorageInternal.promiseInit = CloudStorage.init();