2014-05-12 06:53:00 +00:00
|
|
|
const {Cc,Ci,Cu,Cr} = require("chrome");
|
2013-08-20 08:06:00 +00:00
|
|
|
const ObservableObject = require("devtools/shared/observable-object");
|
2014-04-22 19:55:01 +00:00
|
|
|
const promise = require("devtools/toolkit/deprecated-sync-thenables");
|
2013-08-20 08:06:00 +00:00
|
|
|
|
2014-02-26 04:22:05 +00:00
|
|
|
const {EventEmitter} = Cu.import("resource://gre/modules/devtools/event-emitter.js");
|
2013-09-10 11:26:00 +00:00
|
|
|
const {generateUUID} = Cc['@mozilla.org/uuid-generator;1'].getService(Ci.nsIUUIDGenerator);
|
2014-05-12 06:53:00 +00:00
|
|
|
const {FileUtils} = Cu.import("resource://gre/modules/FileUtils.jsm");
|
2013-08-20 08:06:00 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* IndexedDB wrapper that just save project objects
|
|
|
|
*
|
|
|
|
* The only constraint is that project objects have to have
|
|
|
|
* a unique `location` object.
|
|
|
|
*/
|
|
|
|
|
|
|
|
const global = this;
|
|
|
|
const IDB = {
|
|
|
|
_db: null,
|
|
|
|
|
|
|
|
open: function () {
|
|
|
|
let deferred = promise.defer();
|
|
|
|
|
|
|
|
var idbManager = Cc["@mozilla.org/dom/indexeddb/manager;1"]
|
|
|
|
.getService(Ci.nsIIndexedDatabaseManager);
|
|
|
|
idbManager.initWindowless(global);
|
|
|
|
|
|
|
|
let request = global.indexedDB.open("AppProjects", 5);
|
|
|
|
request.onerror = function(event) {
|
|
|
|
deferred.reject("Unable to open AppProjects indexedDB. " +
|
|
|
|
"Error code: " + event.target.errorCode);
|
|
|
|
};
|
|
|
|
request.onupgradeneeded = function(event) {
|
|
|
|
let db = event.target.result;
|
|
|
|
db.createObjectStore("projects", { keyPath: "location" });
|
|
|
|
};
|
|
|
|
|
|
|
|
request.onsuccess = function() {
|
|
|
|
let db = IDB._db = request.result;
|
|
|
|
let objectStore = db.transaction("projects").objectStore("projects");
|
|
|
|
let projects = []
|
2014-05-12 06:53:00 +00:00
|
|
|
let toRemove = [];
|
2013-08-20 08:06:00 +00:00
|
|
|
objectStore.openCursor().onsuccess = function(event) {
|
|
|
|
let cursor = event.target.result;
|
|
|
|
if (cursor) {
|
2013-10-09 13:31:34 +00:00
|
|
|
if (cursor.value.location) {
|
2014-05-12 06:53:00 +00:00
|
|
|
|
2013-10-09 13:31:34 +00:00
|
|
|
// We need to make sure this object has a `.location` property.
|
|
|
|
// The UI depends on this property.
|
|
|
|
// This should not be needed as we make sure to register valid
|
|
|
|
// projects, but in the past (before bug 924568), we might have
|
|
|
|
// registered invalid objects.
|
2014-05-12 06:53:00 +00:00
|
|
|
|
|
|
|
|
|
|
|
// We also want to make sure the location is valid.
|
|
|
|
// If the location doesn't exist, we remove the project.
|
|
|
|
|
|
|
|
try {
|
|
|
|
let file = FileUtils.File(cursor.value.location);
|
|
|
|
if (file.exists()) {
|
|
|
|
projects.push(cursor.value);
|
|
|
|
} else {
|
|
|
|
toRemove.push(cursor.value.location);
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
if (e.result == Cr.NS_ERROR_FILE_UNRECOGNIZED_PATH) {
|
|
|
|
// A URL
|
|
|
|
projects.push(cursor.value);
|
|
|
|
}
|
|
|
|
}
|
2013-10-09 13:31:34 +00:00
|
|
|
}
|
2013-08-20 08:06:00 +00:00
|
|
|
cursor.continue();
|
|
|
|
} else {
|
2014-05-12 06:53:00 +00:00
|
|
|
let removePromises = [];
|
|
|
|
for (let location of toRemove) {
|
|
|
|
removePromises.push(IDB.remove(location));
|
|
|
|
}
|
|
|
|
promise.all(removePromises).then(() => {
|
|
|
|
deferred.resolve(projects);
|
|
|
|
});
|
2013-08-20 08:06:00 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
return deferred.promise;
|
|
|
|
},
|
|
|
|
|
|
|
|
add: function(project) {
|
|
|
|
let deferred = promise.defer();
|
|
|
|
|
2013-10-09 13:31:34 +00:00
|
|
|
if (!project.location) {
|
|
|
|
// We need to make sure this object has a `.location` property.
|
|
|
|
deferred.reject("Missing location property on project object.");
|
|
|
|
} else {
|
|
|
|
let transaction = IDB._db.transaction(["projects"], "readwrite");
|
|
|
|
let objectStore = transaction.objectStore("projects");
|
|
|
|
let request = objectStore.add(project);
|
|
|
|
request.onerror = function(event) {
|
|
|
|
deferred.reject("Unable to add project to the AppProjects indexedDB: " +
|
|
|
|
this.error.name + " - " + this.error.message );
|
|
|
|
};
|
|
|
|
request.onsuccess = function() {
|
|
|
|
deferred.resolve();
|
|
|
|
};
|
|
|
|
}
|
2013-08-20 08:06:00 +00:00
|
|
|
|
|
|
|
return deferred.promise;
|
|
|
|
},
|
|
|
|
|
2013-09-26 18:07:44 +00:00
|
|
|
update: function(project) {
|
|
|
|
let deferred = promise.defer();
|
|
|
|
|
2014-05-12 06:53:00 +00:00
|
|
|
// Clone object to make it storable by IndexedDB.
|
|
|
|
// Projects are proxified objects (for the template
|
|
|
|
// mechanismn in the first version of the App Manager).
|
|
|
|
// This will change in the future.
|
|
|
|
project = JSON.parse(JSON.stringify(project));
|
|
|
|
|
2013-09-26 18:07:44 +00:00
|
|
|
var transaction = IDB._db.transaction(["projects"], "readwrite");
|
|
|
|
var objectStore = transaction.objectStore("projects");
|
|
|
|
var request = objectStore.put(project);
|
|
|
|
request.onerror = function(event) {
|
|
|
|
deferred.reject("Unable to update project to the AppProjects indexedDB: " +
|
|
|
|
this.error.name + " - " + this.error.message );
|
|
|
|
};
|
|
|
|
request.onsuccess = function() {
|
|
|
|
deferred.resolve();
|
|
|
|
};
|
|
|
|
|
|
|
|
return deferred.promise;
|
|
|
|
},
|
|
|
|
|
2013-08-20 08:06:00 +00:00
|
|
|
remove: function(location) {
|
|
|
|
let deferred = promise.defer();
|
|
|
|
|
|
|
|
let request = IDB._db.transaction(["projects"], "readwrite")
|
|
|
|
.objectStore("projects")
|
|
|
|
.delete(location);
|
|
|
|
request.onsuccess = function(event) {
|
|
|
|
deferred.resolve();
|
|
|
|
};
|
|
|
|
request.onerror = function() {
|
|
|
|
deferred.reject("Unable to delete project to the AppProjects indexedDB: " +
|
|
|
|
this.error.name + " - " + this.error.message );
|
|
|
|
};
|
|
|
|
|
|
|
|
return deferred.promise;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const store = new ObservableObject({ projects:[] });
|
|
|
|
|
2013-09-25 02:22:08 +00:00
|
|
|
let loadDeferred = promise.defer();
|
|
|
|
|
2013-08-20 08:06:00 +00:00
|
|
|
IDB.open().then(function (projects) {
|
|
|
|
store.object.projects = projects;
|
|
|
|
AppProjects.emit("ready", store.object.projects);
|
2013-09-25 02:22:08 +00:00
|
|
|
loadDeferred.resolve();
|
2013-08-20 08:06:00 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
const AppProjects = {
|
2013-09-25 02:22:08 +00:00
|
|
|
load: function() {
|
|
|
|
return loadDeferred.promise;
|
|
|
|
},
|
|
|
|
|
2013-08-20 08:06:00 +00:00
|
|
|
addPackaged: function(folder) {
|
2014-05-12 06:53:00 +00:00
|
|
|
let file = FileUtils.File(folder.path);
|
|
|
|
if (!file.exists()) {
|
|
|
|
return promise.reject("path doesn't exist");
|
|
|
|
}
|
|
|
|
let existingProject = this.get(folder.path);
|
|
|
|
if (existingProject) {
|
|
|
|
return promise.reject("Already added");
|
|
|
|
}
|
2013-08-20 08:06:00 +00:00
|
|
|
let project = {
|
|
|
|
type: "packaged",
|
2013-09-10 11:26:00 +00:00
|
|
|
location: folder.path,
|
|
|
|
// We need a unique id, that is the app origin,
|
|
|
|
// in order to identify the app when being installed on the device.
|
|
|
|
// The packaged app local path is a valid id, but only on the client.
|
|
|
|
// This origin will be used to generate the true id of an app:
|
|
|
|
// its manifest URL.
|
2013-09-26 18:07:44 +00:00
|
|
|
// If the app ends up specifying an explicit origin in its manifest,
|
|
|
|
// we will override this random UUID on app install.
|
2013-09-10 11:26:00 +00:00
|
|
|
packagedAppOrigin: generateUUID().toString().slice(1, -1)
|
2013-08-20 08:06:00 +00:00
|
|
|
};
|
|
|
|
return IDB.add(project).then(function () {
|
|
|
|
store.object.projects.push(project);
|
|
|
|
// return the added objects (proxified)
|
|
|
|
return store.object.projects[store.object.projects.length - 1];
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
addHosted: function(manifestURL) {
|
2014-05-12 06:53:00 +00:00
|
|
|
let existingProject = this.get(manifestURL);
|
|
|
|
if (existingProject) {
|
|
|
|
return promise.reject("Already added");
|
|
|
|
}
|
2013-08-20 08:06:00 +00:00
|
|
|
let project = {
|
|
|
|
type: "hosted",
|
|
|
|
location: manifestURL
|
|
|
|
};
|
|
|
|
return IDB.add(project).then(function () {
|
|
|
|
store.object.projects.push(project);
|
|
|
|
// return the added objects (proxified)
|
|
|
|
return store.object.projects[store.object.projects.length - 1];
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2013-09-26 18:07:44 +00:00
|
|
|
update: function (project) {
|
2014-05-12 06:53:00 +00:00
|
|
|
return IDB.update(project);
|
2013-09-26 18:07:44 +00:00
|
|
|
},
|
|
|
|
|
2013-08-20 08:06:00 +00:00
|
|
|
remove: function(location) {
|
|
|
|
return IDB.remove(location).then(function () {
|
|
|
|
let projects = store.object.projects;
|
|
|
|
for (let i = 0; i < projects.length; i++) {
|
|
|
|
if (projects[i].location == location) {
|
|
|
|
projects.splice(i, 1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw new Error("Unable to find project in AppProjects store");
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
get: function(location) {
|
|
|
|
let projects = store.object.projects;
|
|
|
|
for (let i = 0; i < projects.length; i++) {
|
|
|
|
if (projects[i].location == location) {
|
|
|
|
return projects[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
|
|
|
store: store
|
|
|
|
};
|
|
|
|
|
|
|
|
EventEmitter.decorate(AppProjects);
|
|
|
|
|
|
|
|
exports.AppProjects = AppProjects;
|
|
|
|
|