2023-02-06 02:49:59 +00:00
|
|
|
describe("Retractions", function () {
|
2019-06-10 06:37:54 +00:00
|
|
|
var userLibraryID;
|
|
|
|
var win;
|
|
|
|
var zp;
|
|
|
|
var checkQueueItemsStub;
|
2023-09-03 22:43:57 +00:00
|
|
|
var retractedDOI = '10.1056/NEJMoa1200303'; // mixed case
|
2019-06-10 06:37:54 +00:00
|
|
|
|
|
|
|
before(async function () {
|
2024-04-08 07:11:23 +00:00
|
|
|
// TEMP: Temporarily disabled in CI due to failures in fx115
|
|
|
|
if (Zotero.automatedTest) {
|
|
|
|
this.skip();
|
|
|
|
}
|
|
|
|
|
2019-06-10 06:37:54 +00:00
|
|
|
userLibraryID = Zotero.Libraries.userLibraryID;
|
|
|
|
win = await loadZoteroPane();
|
|
|
|
zp = win.ZoteroPane;
|
2019-06-07 05:13:42 +00:00
|
|
|
|
2020-05-27 12:44:25 +00:00
|
|
|
await Zotero.Retractions.updateFromServer();
|
|
|
|
|
2019-06-10 06:37:54 +00:00
|
|
|
// Remove debouncing on checkQueuedItems()
|
|
|
|
checkQueueItemsStub = sinon.stub(Zotero.Retractions, 'checkQueuedItems').callsFake(() => {
|
|
|
|
return Zotero.Retractions._checkQueuedItemsInternal();
|
2019-06-07 05:13:42 +00:00
|
|
|
});
|
2019-06-10 06:37:54 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
beforeEach(async function () {
|
|
|
|
var ids = await Zotero.DB.columnQueryAsync("SELECT itemID FROM retractedItems");
|
|
|
|
if (ids.length) {
|
|
|
|
await Zotero.Items.erase(ids);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(async function () {
|
|
|
|
win.document.getElementById('retracted-items-close').click();
|
|
|
|
checkQueueItemsStub.resetHistory();
|
|
|
|
});
|
|
|
|
|
|
|
|
after(async function () {
|
2024-04-08 07:11:23 +00:00
|
|
|
// TEMP
|
|
|
|
if (Zotero.automatedTest) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-10 06:37:54 +00:00
|
|
|
win.close();
|
|
|
|
checkQueueItemsStub.restore();
|
2019-06-07 05:13:42 +00:00
|
|
|
|
2019-06-10 06:37:54 +00:00
|
|
|
var ids = await Zotero.DB.columnQueryAsync("SELECT itemID FROM retractedItems");
|
|
|
|
if (ids.length) {
|
|
|
|
await Zotero.Items.erase(ids);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
async function createRetractedItem(options = {}) {
|
|
|
|
var o = {
|
|
|
|
itemType: 'journalArticle'
|
|
|
|
};
|
|
|
|
Object.assign(o, options);
|
|
|
|
var item = createUnsavedDataObject('item', o);
|
|
|
|
item.setField('DOI', retractedDOI);
|
2020-06-25 22:50:17 +00:00
|
|
|
if (Zotero.DB.inTransaction()) {
|
|
|
|
await item.save();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
await item.saveTx();
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!checkQueueItemsStub.called) {
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
}
|
|
|
|
await checkQueueItemsStub.returnValues[0];
|
|
|
|
checkQueueItemsStub.resetHistory();
|
|
|
|
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function createRetractedItemWithExtraDOI(options = {}) {
|
|
|
|
var o = {
|
|
|
|
itemType: 'journalArticle'
|
|
|
|
};
|
|
|
|
Object.assign(o, options);
|
|
|
|
var item = createUnsavedDataObject('item', o);
|
|
|
|
item.setField('extra', 'DOI: ' + retractedDOI);
|
|
|
|
if (Zotero.DB.inTransaction()) {
|
2019-06-10 06:37:54 +00:00
|
|
|
await item.save();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
await item.saveTx();
|
|
|
|
}
|
2019-06-07 05:13:42 +00:00
|
|
|
|
2019-06-10 06:37:54 +00:00
|
|
|
while (!checkQueueItemsStub.called) {
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
}
|
|
|
|
await checkQueueItemsStub.returnValues[0];
|
|
|
|
checkQueueItemsStub.resetHistory();
|
|
|
|
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
2019-06-12 01:24:17 +00:00
|
|
|
function bannerShown() {
|
|
|
|
var container = win.document.getElementById('retracted-items-container');
|
|
|
|
if (container.getAttribute('collapsed') == 'true') {
|
|
|
|
return false;
|
2019-06-10 06:37:54 +00:00
|
|
|
}
|
2019-06-12 01:24:17 +00:00
|
|
|
if (!container.hasAttribute('collapsed')) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
throw new Error("'collapsed' attribute not found");
|
|
|
|
}
|
|
|
|
|
2019-06-12 05:05:49 +00:00
|
|
|
|
2019-06-19 10:41:33 +00:00
|
|
|
describe("#updateFromServer()", function () {
|
|
|
|
var server;
|
|
|
|
var baseURL;
|
|
|
|
|
|
|
|
before(function () {
|
|
|
|
Zotero.HTTP.mock = sinon.FakeXMLHttpRequest;
|
|
|
|
baseURL = ZOTERO_CONFIG.API_URL + 'retractions/';
|
|
|
|
});
|
|
|
|
|
|
|
|
beforeEach(function () {
|
|
|
|
server = sinon.fakeServer.create();
|
|
|
|
server.autoRespond = true;
|
|
|
|
});
|
|
|
|
|
|
|
|
after(async function () {
|
|
|
|
Zotero.HTTP.mock = null;
|
|
|
|
// Restore the real list from the server. We could just mock it as part of the suite.
|
|
|
|
await Zotero.Retractions.updateFromServer();
|
|
|
|
});
|
|
|
|
|
|
|
|
/*it("shouldn't show banner or virtual collection for already flagged items on list update", async function () {
|
|
|
|
await Zotero.Retractions.updateFromServer();
|
|
|
|
});*/
|
|
|
|
|
|
|
|
it("should remove retraction flag from items that no longer match prefix list", async function () {
|
|
|
|
var doi = '10.1234/abcde';
|
|
|
|
var hash = Zotero.Utilities.Internal.sha1(doi);
|
|
|
|
var prefix = hash.substr(0, 5);
|
|
|
|
var lines = [
|
|
|
|
Zotero.Retractions.TYPE_DOI + prefix + ' 12345\n',
|
|
|
|
Zotero.Retractions.TYPE_DOI + 'aaaaa 23456\n'
|
|
|
|
];
|
|
|
|
|
|
|
|
var listCount = 0;
|
|
|
|
var searchCount = 0;
|
|
|
|
server.respond(function (req) {
|
|
|
|
if (req.method == 'GET' && req.url == baseURL + 'list') {
|
|
|
|
listCount++;
|
|
|
|
if (listCount == 1) {
|
|
|
|
req.respond(
|
|
|
|
200,
|
|
|
|
{
|
|
|
|
'Content-Type': 'text/plain',
|
|
|
|
'ETag': 'abcdefg'
|
|
|
|
},
|
|
|
|
lines.join('')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else if (listCount == 2) {
|
|
|
|
req.respond(
|
|
|
|
200,
|
|
|
|
{
|
|
|
|
'Content-Type': 'text/plain',
|
|
|
|
'ETag': 'bcdefgh'
|
|
|
|
},
|
|
|
|
lines[1]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (req.method == 'POST' && req.url == baseURL + 'search') {
|
|
|
|
searchCount++;
|
|
|
|
if (searchCount == 1) {
|
|
|
|
req.respond(
|
|
|
|
200,
|
|
|
|
{
|
|
|
|
'Content-Type': 'application/json'
|
|
|
|
},
|
|
|
|
JSON.stringify([
|
|
|
|
{
|
|
|
|
doi: hash,
|
|
|
|
retractionDOI: '10.1234/bcdef',
|
2020-06-22 06:28:05 +00:00
|
|
|
date: '2019-01-02',
|
|
|
|
reasons: [
|
|
|
|
"Error in Data"
|
|
|
|
],
|
|
|
|
urls: []
|
2019-06-19 10:41:33 +00:00
|
|
|
}
|
|
|
|
])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
await Zotero.Retractions.updateFromServer();
|
|
|
|
|
|
|
|
// Create item with DOI from list
|
|
|
|
var promise = waitForItemEvent('refresh');
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
|
|
|
await promise;
|
|
|
|
|
|
|
|
assert.isTrue(Zotero.Retractions.isRetracted(item));
|
|
|
|
|
|
|
|
// Make a second request, with the entry removed
|
|
|
|
promise = waitForItemEvent('refresh');
|
|
|
|
await Zotero.Retractions.updateFromServer();
|
|
|
|
await promise;
|
|
|
|
|
|
|
|
assert.isFalse(Zotero.Retractions.isRetracted(item));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2019-07-03 05:23:02 +00:00
|
|
|
describe("#shouldShowCitationWarning()", function () {
|
|
|
|
it("should return false if citation warning is hidden", async function () {
|
|
|
|
var item = await createRetractedItem();
|
|
|
|
assert.isTrue(Zotero.Retractions.shouldShowCitationWarning(item));
|
|
|
|
await Zotero.Retractions.disableCitationWarningsForItem(item);
|
|
|
|
assert.isFalse(Zotero.Retractions.shouldShowCitationWarning(item));
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should return false if retraction is hidden", async function () {
|
|
|
|
var item = await createRetractedItem();
|
|
|
|
assert.isTrue(Zotero.Retractions.shouldShowCitationWarning(item));
|
|
|
|
await Zotero.Retractions.hideRetraction(item);
|
|
|
|
assert.isFalse(Zotero.Retractions.shouldShowCitationWarning(item));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2019-06-12 05:05:49 +00:00
|
|
|
describe("#getRetractionsFromJSON()", function () {
|
|
|
|
it("should identify object with retracted DOI", async function () {
|
|
|
|
var spy = sinon.spy(Zotero.HTTP, 'request');
|
|
|
|
var json = [
|
|
|
|
{
|
|
|
|
|
|
|
|
},
|
|
|
|
{
|
|
|
|
DOI: retractedDOI
|
|
|
|
},
|
|
|
|
{
|
|
|
|
DOI: '10.1234/abcd'
|
|
|
|
}
|
|
|
|
];
|
|
|
|
|
|
|
|
var indexes = await Zotero.Retractions.getRetractionsFromJSON(json);
|
|
|
|
assert.sameMembers(indexes, [1]);
|
|
|
|
assert.equal(spy.callCount, 1);
|
|
|
|
|
|
|
|
indexes = await Zotero.Retractions.getRetractionsFromJSON(json);
|
|
|
|
assert.sameMembers(indexes, [1]);
|
|
|
|
// Result should've been cached, so we should have it without another API request
|
|
|
|
assert.equal(spy.callCount, 1);
|
|
|
|
|
|
|
|
spy.restore();
|
|
|
|
});
|
2020-06-25 22:50:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
it("should identify object with retracted DOI in Extra", async function () {
|
|
|
|
var json = [
|
|
|
|
{
|
|
|
|
extra: `DOI: ${retractedDOI}`
|
|
|
|
}
|
|
|
|
];
|
|
|
|
|
|
|
|
var indexes = await Zotero.Retractions.getRetractionsFromJSON(json);
|
|
|
|
assert.sameMembers(indexes, [0]);
|
|
|
|
});
|
2020-09-24 02:49:25 +00:00
|
|
|
|
|
|
|
it("should identify object with retracted DOI on subsequent line in Extra", async function () {
|
|
|
|
var json = [
|
|
|
|
{
|
|
|
|
extra: `Foo: Bar\nDOI: ${retractedDOI}`
|
|
|
|
}
|
|
|
|
];
|
|
|
|
|
|
|
|
var indexes = await Zotero.Retractions.getRetractionsFromJSON(json);
|
|
|
|
assert.sameMembers(indexes, [0]);
|
|
|
|
});
|
2019-06-12 05:05:49 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
|
2019-06-12 01:24:17 +00:00
|
|
|
describe("Notification Banner", function () {
|
2019-06-10 06:37:54 +00:00
|
|
|
it("should show banner when retracted item is added", async function () {
|
|
|
|
assert.isFalse(bannerShown());
|
|
|
|
await createRetractedItem();
|
2023-04-28 05:53:29 +00:00
|
|
|
do {
|
|
|
|
await delay(10);
|
|
|
|
}
|
|
|
|
while (!bannerShown());
|
2019-06-10 06:37:54 +00:00
|
|
|
});
|
|
|
|
|
2020-06-25 22:50:17 +00:00
|
|
|
it("should show banner when retracted item with DOI in Extra is added", async function () {
|
|
|
|
assert.isFalse(bannerShown());
|
|
|
|
await createRetractedItemWithExtraDOI();
|
2023-04-28 05:53:29 +00:00
|
|
|
do {
|
|
|
|
await delay(10);
|
|
|
|
}
|
|
|
|
while (!bannerShown());
|
2020-06-25 22:50:17 +00:00
|
|
|
});
|
|
|
|
|
2019-06-10 06:37:54 +00:00
|
|
|
it("shouldn't show banner when item in trash is added", async function () {
|
2023-02-06 02:49:59 +00:00
|
|
|
await createRetractedItem({ deleted: true });
|
2023-04-28 05:53:29 +00:00
|
|
|
await delay(50);
|
2019-06-10 06:37:54 +00:00
|
|
|
assert.isFalse(bannerShown());
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("virtual collection", function () {
|
|
|
|
it("should show/hide Retracted Items collection when a retracted item is found/erased", async function () {
|
|
|
|
// Create item
|
|
|
|
var item = await createRetractedItem();
|
|
|
|
assert.ok(zp.collectionsView.getRowIndexByID("R" + userLibraryID));
|
|
|
|
|
|
|
|
// Erase item
|
|
|
|
var promise = waitForItemEvent('refresh');
|
|
|
|
await item.eraseTx();
|
|
|
|
await promise;
|
|
|
|
assert.isFalse(zp.collectionsView.getRowIndexByID("R" + userLibraryID));
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should unhide Retracted Items collection when retracted item is found", async function () {
|
|
|
|
await createRetractedItem();
|
|
|
|
|
|
|
|
// Hide collection
|
|
|
|
await zp.setVirtual(userLibraryID, 'retracted', false);
|
|
|
|
|
|
|
|
// Add another retracted item, which should unhide it
|
|
|
|
await createRetractedItem();
|
|
|
|
assert.ok(zp.collectionsView.getRowIndexByID("R" + userLibraryID));
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should hide Retracted Items collection when last retracted item is moved to trash", async function () {
|
|
|
|
var rowID = "R" + userLibraryID;
|
|
|
|
|
|
|
|
// Create item
|
|
|
|
var item = await createRetractedItem();
|
|
|
|
assert.ok(zp.collectionsView.getRowIndexByID(rowID));
|
|
|
|
|
|
|
|
// Select Retracted Items collection
|
|
|
|
await zp.collectionsView.selectByID(rowID);
|
|
|
|
await waitForItemsLoad(win);
|
|
|
|
|
|
|
|
// Erase item
|
|
|
|
item.deleted = true;
|
|
|
|
await item.saveTx();
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
// Retracted Items should be gone
|
|
|
|
assert.isFalse(zp.collectionsView.getRowIndexByID(rowID));
|
|
|
|
// And My Library should be selected
|
|
|
|
assert.equal(zp.collectionsView.selectedTreeRow.id, "L" + userLibraryID);
|
|
|
|
});
|
|
|
|
|
2019-07-03 05:23:02 +00:00
|
|
|
it("should hide Retracted Items collection when last retracted item is marked as hidden", async function () {
|
|
|
|
var rowID = "R" + userLibraryID;
|
|
|
|
|
|
|
|
// Create item
|
|
|
|
var item = await createRetractedItem();
|
|
|
|
assert.ok(zp.collectionsView.getRowIndexByID(rowID));
|
|
|
|
|
|
|
|
// Select Retracted Items collection
|
|
|
|
await zp.collectionsView.selectByID(rowID);
|
|
|
|
await waitForItemsLoad(win);
|
|
|
|
|
|
|
|
await Zotero.Retractions.hideRetraction(item);
|
|
|
|
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
// Retracted Items should be gone
|
|
|
|
assert.isFalse(zp.collectionsView.getRowIndexByID(rowID));
|
|
|
|
// And My Library should be selected
|
|
|
|
assert.equal(zp.collectionsView.selectedTreeRow.id, "L" + userLibraryID);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("shouldn't hide Retracted Items collection when last retracted item is marked to not show a citation warning", async function () {
|
|
|
|
var rowID = "R" + userLibraryID;
|
|
|
|
|
|
|
|
// Create item
|
|
|
|
var item = await createRetractedItem();
|
|
|
|
assert.ok(zp.collectionsView.getRowIndexByID(rowID));
|
|
|
|
|
|
|
|
// Select Retracted Items collection
|
|
|
|
await zp.collectionsView.selectByID(rowID);
|
|
|
|
await waitForItemsLoad(win);
|
|
|
|
|
|
|
|
await Zotero.Retractions.disableCitationWarningsForItem(item);
|
|
|
|
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
// Should still be showing
|
|
|
|
assert.ok(zp.collectionsView.getRowIndexByID("R" + userLibraryID));
|
|
|
|
});
|
|
|
|
|
2019-06-10 06:37:54 +00:00
|
|
|
it("should show Retracted Items collection when retracted item is restored from trash", async function () {
|
|
|
|
// Create trashed item
|
|
|
|
var item = await createRetractedItem({ deleted: true });
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
assert.isFalse(zp.collectionsView.getRowIndexByID("R" + userLibraryID));
|
|
|
|
|
|
|
|
// Restore item
|
|
|
|
item.deleted = false;
|
|
|
|
await item.saveTx();
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
assert.ok(zp.collectionsView.getRowIndexByID("R" + userLibraryID));
|
2019-06-07 05:13:42 +00:00
|
|
|
});
|
|
|
|
});
|
2019-06-12 01:24:17 +00:00
|
|
|
|
|
|
|
describe("retractions.enabled", function () {
|
|
|
|
beforeEach(function () {
|
|
|
|
Zotero.Prefs.clear('retractions.enabled');
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should hide virtual collection and banner when false", async function () {
|
|
|
|
var item = await createRetractedItem();
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
var itemRetractionBox = win.document.getElementById('retraction-box');
|
|
|
|
assert.isFalse(itemRetractionBox.hidden);
|
|
|
|
|
|
|
|
var spies = [
|
|
|
|
sinon.spy(Zotero.Retractions, '_removeAllEntries'),
|
2019-07-04 11:58:56 +00:00
|
|
|
sinon.spy(Zotero.Retractions, 'isRetracted')
|
2019-06-12 01:24:17 +00:00
|
|
|
];
|
|
|
|
Zotero.Prefs.set('retractions.enabled', false);
|
|
|
|
|
|
|
|
while (!spies[0].called || !spies[1].called) {
|
|
|
|
await Zotero.Promise.delay(50);
|
|
|
|
}
|
|
|
|
await spies[0].returnValues[0];
|
2023-02-06 02:49:59 +00:00
|
|
|
await spies[1].returnValues[0];
|
2019-06-12 01:24:17 +00:00
|
|
|
spies.forEach(spy => spy.restore());
|
|
|
|
|
|
|
|
assert.isFalse(Zotero.Retractions.isRetracted(item));
|
|
|
|
assert.isFalse(zp.collectionsView.getRowIndexByID("R" + userLibraryID));
|
|
|
|
assert.isFalse(bannerShown());
|
|
|
|
|
|
|
|
assert.isTrue(itemRetractionBox.hidden);
|
|
|
|
|
|
|
|
await item.eraseTx();
|
|
|
|
});
|
|
|
|
});
|
2023-02-06 02:49:59 +00:00
|
|
|
});
|