
- Protocol handler extensions can now handle promises and can also make data available as it's ready instead of all at once (e.g., reports now output one entry at a time) - zotero:// URL syntaxes are now more consistent and closer to the web API (old URLs should work, but some may currently be broken) Also: - Code to generate server API, currently available for testing via zotero://data URLs but eventually moving to HTTP -- zotero://data URLs match web API URLs, with a different prefix for the personal library (/library vs. /users/12345) - Miscellaneous fixes to data objects Under the hood: - Extensions now return an AsyncChannel, which is an nsIChannel implementation that takes a promise-yielding generator that returns a string, nsIAsyncInputStream, or file that will be used for the channel's data - New function Zotero.Utilities.Internal.getAsyncInputStream() takes a generator that yields either promises or strings and returns an async input stream filled with the yielded strings - Zotero.Router parsers URLs and extract parameters - Zotero.Item.toResponseJSON()
191 lines
5 KiB
JavaScript
191 lines
5 KiB
JavaScript
/*
|
|
***** BEGIN LICENSE BLOCK *****
|
|
|
|
Copyright © 2014 Center for History and New Media
|
|
George Mason University, Fairfax, Virginia, USA
|
|
http://zotero.org
|
|
|
|
This file is part of Zotero.
|
|
|
|
Zotero is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Affero General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
Zotero is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Affero General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
along with Zotero. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
***** END LICENSE BLOCK *****
|
|
*/
|
|
|
|
Zotero.API = {
|
|
parseParams: function (params) {
|
|
if (params.groupID) {
|
|
params.libraryID = Zotero.Groups.getLibraryIDFromGroupID(params.groupID);
|
|
}
|
|
|
|
if (typeof params.itemKey == 'string') {
|
|
params.itemKey = params.itemKey.split(',');
|
|
}
|
|
},
|
|
|
|
|
|
getResultsFromParams: Zotero.Promise.coroutine(function* (params) {
|
|
var results;
|
|
switch (params.scopeObject) {
|
|
case 'collections':
|
|
if (params.scopeObjectKey) {
|
|
var col = yield Zotero.Collections.getByLibraryAndKeyAsync(
|
|
params.libraryID, params.scopeObjectKey
|
|
);
|
|
}
|
|
else {
|
|
var col = yield Zotero.Collections.getAsync(params.scopeObjectID);
|
|
}
|
|
if (!col) {
|
|
throw new Error('Invalid collection ID or key');
|
|
}
|
|
yield col.loadChildItems();
|
|
results = col.getChildItems();
|
|
break;
|
|
|
|
case 'searches':
|
|
if (params.scopeObjectKey) {
|
|
var s = yield Zotero.Searches.getByLibraryAndKeyAsync(
|
|
params.libraryID, params.scopeObjectKey
|
|
);
|
|
}
|
|
else {
|
|
var s = yield Zotero.Searches.getAsync(params.scopeObjectID);
|
|
}
|
|
if (!s) {
|
|
throw new Error('Invalid search ID or key');
|
|
}
|
|
|
|
// FIXME: Hack to exclude group libraries for now
|
|
var s2 = new Zotero.Search();
|
|
s2.setScope(s);
|
|
var groups = Zotero.Groups.getAll();
|
|
for each(var group in groups) {
|
|
yield s2.addCondition('libraryID', 'isNot', group.libraryID);
|
|
}
|
|
var ids = yield s2.search();
|
|
break;
|
|
|
|
default:
|
|
if (params.scopeObject) {
|
|
throw new Error("Invalid scope object '" + params.scopeObject + "'");
|
|
}
|
|
|
|
if (params.itemKey) {
|
|
var s = new Zotero.Search;
|
|
yield s.addCondition('libraryID', 'is', params.libraryID);
|
|
yield s.addCondition('blockStart');
|
|
for (let i=0; i<params.itemKey.length; i++) {
|
|
let itemKey = params.itemKey[i];
|
|
yield s.addCondition('key', 'is', itemKey);
|
|
}
|
|
yield s.addCondition('blockEnd');
|
|
var ids = yield s.search();
|
|
}
|
|
else {
|
|
// Display all items
|
|
var s = new Zotero.Search();
|
|
yield s.addCondition('libraryID', 'is', params.libraryID);
|
|
yield s.addCondition('noChildren', 'true');
|
|
var ids = yield s.search();
|
|
}
|
|
}
|
|
|
|
if (results) {
|
|
// Filter results by item key
|
|
if (params.itemKey) {
|
|
results = results.filter(function (result) {
|
|
return params.itemKey.indexOf(result.key) !== -1;
|
|
});
|
|
}
|
|
}
|
|
else if (ids) {
|
|
// Filter results by item key
|
|
if (params.itemKey) {
|
|
ids = ids.filter(function (id) {
|
|
var [libraryID, key] = Zotero.Items.getLibraryAndKeyFromID(id);
|
|
return params.itemKey.indexOf(key) !== -1;
|
|
});
|
|
}
|
|
results = yield Zotero.Items.getAsync(ids);
|
|
}
|
|
|
|
return results;
|
|
}),
|
|
|
|
|
|
getLibraryPrefix: function (libraryID) {
|
|
return libraryID
|
|
? 'groups/' + Zotero.Groups.getGroupIDFromLibraryID(libraryID)
|
|
: 'library';
|
|
}
|
|
};
|
|
|
|
Zotero.API.Data = {
|
|
/**
|
|
* Parse a relative URI path and return parameters for the request
|
|
*/
|
|
parsePath: function (path) {
|
|
var params = {};
|
|
var router = new Zotero.Router(params);
|
|
|
|
// Top-level objects
|
|
router.add('library/:controller/top', function () {
|
|
params.libraryID = 0;
|
|
params.subset = 'top';
|
|
});
|
|
router.add('groups/:groupID/:controller/top', function () {
|
|
params.subset = 'top';
|
|
});
|
|
|
|
router.add('library/:scopeObject/:scopeObjectKey/items/:objectKey/:subset', function () {
|
|
params.libraryID = 0;
|
|
params.controller = 'items';
|
|
});
|
|
router.add('groups/:groupID/:scopeObject/:scopeObjectKey/items/:objectKey/:subset', function () {
|
|
params.controller = 'items';
|
|
});
|
|
|
|
// All objects
|
|
router.add('library/:controller', function () {
|
|
params.libraryID = 0;
|
|
});
|
|
router.add('groups/:groupID/:controller', function () {});
|
|
|
|
var parsed = router.run(path);
|
|
if (!parsed || !params.controller) {
|
|
throw new Zotero.Router.InvalidPathException(path);
|
|
}
|
|
|
|
if (params.groupID) {
|
|
params.libraryID = Zotero.Groups.getLibraryIDFromGroupID(params.groupID);
|
|
}
|
|
Zotero.Router.Utilities.convertControllerToObjectType(params);
|
|
|
|
return params;
|
|
},
|
|
|
|
|
|
getGenerator: function (path) {
|
|
var params = this.parsePath(path);
|
|
//Zotero.debug(params);
|
|
|
|
return Zotero.DataObjectUtilities.getClassForObjectType(params.objectType)
|
|
.apiDataGenerator(params);
|
|
}
|
|
};
|
|
|
|
|
|
|
|
|