describe("Retractions", function() { var userLibraryID; var win; var zp; var server; var checkQueueItemsStub; var retractedDOI = '10.1016/S0140-6736(97)11096-0'; before(async function () { userLibraryID = Zotero.Libraries.userLibraryID; win = await loadZoteroPane(); zp = win.ZoteroPane; await Zotero.Retractions.updateFromServer(); // Remove debouncing on checkQueuedItems() checkQueueItemsStub = sinon.stub(Zotero.Retractions, 'checkQueuedItems').callsFake(() => { return Zotero.Retractions._checkQueuedItemsInternal(); }); }); 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 () { win.close(); checkQueueItemsStub.restore(); 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); 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()) { await item.save(); } else { await item.saveTx(); } while (!checkQueueItemsStub.called) { await Zotero.Promise.delay(50); } await checkQueueItemsStub.returnValues[0]; checkQueueItemsStub.resetHistory(); return item; } function bannerShown() { var container = win.document.getElementById('retracted-items-container'); if (container.getAttribute('collapsed') == 'true') { return false; } if (!container.hasAttribute('collapsed')) { return true; } throw new Error("'collapsed' attribute not found"); } 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', date: '2019-01-02', reasons: [ "Error in Data" ], urls: [] } ]) ); } } }); 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)); }); }); 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)); }); }); 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(); }); 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]); }); 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]); }); }); describe("Notification Banner", function () { it("should show banner when retracted item is added", async function () { var banner = win.document.getElementById('retracted-items-container'); assert.isFalse(bannerShown()); await createRetractedItem(); assert.isTrue(bannerShown()); }); it("should show banner when retracted item with DOI in Extra is added", async function () { var banner = win.document.getElementById('retracted-items-container'); assert.isFalse(bannerShown()); await createRetractedItemWithExtraDOI(); assert.isTrue(bannerShown()); }); it("shouldn't show banner when item in trash is added", async function () { var item = await createRetractedItem({ deleted: true }); assert.isFalse(bannerShown()); win.document.getElementById('retracted-items-link').click(); while (zp.collectionsView.selectedTreeRow.id != 'L1') { await Zotero.Promise.delay(10); } await waitForItemsLoad(win); var item = await zp.getSelectedItems()[0]; assert.equal(item, item); }); }); 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); }); 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)); }); 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)); }); }); 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'), sinon.spy(Zotero.Retractions, 'isRetracted') ]; Zotero.Prefs.set('retractions.enabled', false); while (!spies[0].called || !spies[1].called) { await Zotero.Promise.delay(50); } await spies[0].returnValues[0]; await spies[1].returnValues[0] 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(); }); }); });