"use strict"; describe("Zotero.Integration", function () { Components.utils.import("resource://gre/modules/XPCOMUtils.jsm"); const INTEGRATION_TYPE_ITEM = 1; const INTEGRATION_TYPE_BIBLIOGRAPHY = 2; const INTEGRATION_TYPE_TEMP = 3; /** * To be used as a reference for Zotero-Word Integration plugins */ var DocumentPluginDummy = {}; /** * The Application class corresponds to a word processing application. */ DocumentPluginDummy.Application = function() { this.doc = new DocumentPluginDummy.Document(); this.primaryFieldType = "Field"; this.secondaryFieldType = "Bookmark"; this.fields = []; }; DocumentPluginDummy.Application.prototype = { /** * Gets the active document. * @returns {DocumentPluginDummy.Document} */ getActiveDocument: function() {return this.doc}, /** * Gets the document by some app-specific identifier. * @param {String|Number} docID */ getDocument: function(docID) {return this.doc}, QueryInterface: function() {return this}, }; /** * The Document class corresponds to a single word processing document. */ DocumentPluginDummy.Document = function() {this.fields = []}; DocumentPluginDummy.Document.prototype = { /** * Displays a dialog in the word processing application * @param {String} dialogText * @param {Number} icon - one of the constants defined in integration.js for dialog icons * @param {Number} buttons - one of the constants defined in integration.js for dialog buttons * @returns {Number} * - Yes: 2, No: 1, Cancel: 0 * - Yes: 1, No: 0 * - Ok: 1, Cancel: 0 * - Ok: 0 */ displayAlert: (dialogText, icon, buttons) => 0, /** * Brings this document to the foreground (if necessary to return after displaying a dialog) */ activate: () => 0, /** * Determines whether a field can be inserted at the current position. * @param {String} fieldType * @returns {Boolean} */ canInsertField: (fieldType) => true, /** * Returns the field in which the cursor resides, or NULL if none. * @param {String} fieldType * @returns {Boolean} */ cursorInField: (fieldType) => false, /** * Get document data property from the current document * @returns {String} */ getDocumentData: function() {return this.data}, /** * Set document data property * @param {String} data */ setDocumentData: function(data) {this.data = data}, /** * Inserts a field at the given position and initializes the field object. * @param {String} fieldType * @param {Integer} noteType * @returns {DocumentPluginDummy.Field} */ insertField: function(fieldType, noteType) { if (typeof noteType != "number") { throw new Error("noteType must be an integer"); } var field = new DocumentPluginDummy.Field(this); this.fields.push(field); return field }, /** * Gets all fields present in the document. * @param {String} fieldType * @returns {DocumentPluginDummy.FieldEnumerator} */ getFields: function(fieldType) {return new DocumentPluginDummy.FieldEnumerator(this)}, /** * Gets all fields present in the document. The observer will receive notifications for two * topics: "fields-progress", with the document as the subject and percent progress as data, and * "fields-available", with an nsISimpleEnumerator of fields as the subject and the length as * data * @param {String} fieldType * @param {nsIObserver} observer */ getFieldsAsync: function(fieldType, observer) { observer.observe(this.getFields(fieldType), 'fields-available', null) }, /** * Sets the bibliography style, overwriting the current values for this document */ setBibliographyStyle: (firstLineIndent, bodyIndent, lineSpacing, entrySpacing, tabStops, tabStopsCount) => 0, /** * Converts all fields in a document to a different fieldType or noteType * @params {DocumentPluginDummy.FieldEnumerator} fields */ convert: (fields, toFieldType, toNoteType, count) => 0, /** * Cleans up the document state and resumes processor for editing */ cleanup: () => 0, /** * Informs the document processor that the operation is complete */ complete: () => 0, }; DocumentPluginDummy.FieldEnumerator = function(doc) {this.doc = doc; this.idx = 0}; DocumentPluginDummy.FieldEnumerator.prototype = { hasMoreElements: function() {return this.idx < this.doc.fields.length;}, getNext: function() {return this.doc.fields[this.idx++]}, QueryInterface: XPCOMUtils.generateQI([Components.interfaces.nsISupports, Components.interfaces.nsISimpleEnumerator]) }; /** * The Field class corresponds to a field containing an individual citation * or bibliography */ DocumentPluginDummy.Field = function(doc) { this.doc = doc; this.code = ''; // This is actually required and current integration code depends on text being non-empty upon insertion. // insertBibliography will fail if there is no placeholder text. this.text = '{Placeholder}'; this.wrappedJSObject = this; }; DocumentPluginDummy.Field.noteIndex = 0; DocumentPluginDummy.Field.prototype = { /** * Deletes this field and its contents. */ delete: function() {this.doc.fields.filter((field) => field !== this)}, /** * Removes this field, but maintains the field's contents. */ removeCode: function() {this.code = ""}, /** * Selects this field. */ select: () => 0, /** * Sets the text inside this field to a specified plain text string or pseudo-RTF formatted text * string. * @param {String} text * @param {Boolean} isRich */ setText: function(text, isRich) {this.text = text}, /** * Gets the text inside this field, preferably with formatting, but potentially without * @returns {String} */ getText: function() {return this.text}, /** * Sets field's code * @param {String} code */ setCode: function(code) {this.code = code}, /** * Gets field's code. * @returns {String} */ getCode: function() {return this.code}, /** * Returns true if this field and the passed field are actually references to the same field. * @param {DocumentPluginDummy.Field} field * @returns {Boolean} */ equals: function(field) {return this == field}, /** * This field's note index, if it is in a footnote or endnote; otherwise zero. * @returns {Number} */ getNoteIndex: () => 0, }; for (let cls of ['Application', 'Document', 'FieldEnumerator', 'Field']) { for (let methodName in DocumentPluginDummy[cls].prototype) { if (methodName !== 'QueryInterface') { let method = DocumentPluginDummy[cls].prototype[methodName]; DocumentPluginDummy[cls].prototype[methodName] = function() { try { Zotero.debug(`DocumentPluginDummy: ${cls}.${methodName} invoked with args ${JSON.stringify(arguments)}`, 2); } catch (e) { Zotero.debug(`DocumentPluginDummy: ${cls}.${methodName} invoked with args ${arguments}`, 2); } var result = method.apply(this, arguments); try { Zotero.debug(`Result: ${JSON.stringify(result)}`, 2); } catch (e) { Zotero.debug(`Result: ${result}`, 2); } return result; } } } } var testItems; var applications = {}; var addEditCitationSpy, displayDialogStub; var styleID = "http://www.zotero.org/styles/cell"; var stylePath = OS.Path.join(getTestDataDirectory().path, 'cell.csl'); var commandList = [ 'addCitation', 'editCitation', 'addEditCitation', 'addBibliography', 'editBibliography', 'addEditBibliography', 'refresh', 'removeCodes', 'setDocPrefs' ]; function execCommand(command, docID) { if (! commandList.includes(command)) { throw new Error(`${command} is not a valid document command`); } if (typeof docID === "undefined") { throw new Error(`docID cannot be undefined`) } Zotero.debug(`execCommand '${command}': ${docID}`, 2); return Zotero.Integration.execCommand("dummy", command, docID); } var dialogResults = { addCitationDialog: {}, quickFormat: {}, integrationDocPrefs: {}, selectItemsDialog: {}, editBibliographyDialog: {} }; function initDoc(docID, options={}) { applications[docID] = new DocumentPluginDummy.Application(); var data = new Zotero.Integration.DocumentData(); data.prefs = { noteType: 0, fieldType: "Field", automaticJournalAbbreviations: true }; data.style = {styleID, locale: 'en-US', hasBibliography: true, bibliographyStyleHasBeenSet: true}; data.sessionID = Zotero.Utilities.randomString(10); Object.assign(data, options); applications[docID].getActiveDocument().setDocumentData(data.serialize()); } function setDefaultIntegrationDocPrefs() { dialogResults.integrationDocPrefs = { style: "http://www.zotero.org/styles/cell", locale: 'en-US', fieldType: 'Field', automaticJournalAbbreviations: false, useEndnotes: 0 }; } setDefaultIntegrationDocPrefs(); function setAddEditItems(items) { if (items.length == undefined) items = [items]; dialogResults.quickFormat = function(dialogName, io) { io.citation.citationItems = items.map(function(item) { item = Zotero.Cite.getItem(item.id); return {id: item.id, uris: item.cslURIs, itemData: item.cslItemData}; }); io._acceptDeferred.resolve(() => {}); }; } before(function* () { yield Zotero.Styles.init(); yield Zotero.Styles.install({file: stylePath}, styleID, true); testItems = []; for (let i = 0; i < 5; i++) { let testItem = yield createDataObject('item', {libraryID: Zotero.Libraries.userLibraryID}); testItem.setField('title', `title${1}`); testItem.setCreator(0, {creatorType: 'author', name: `Author No${i}`}); testItems.push(testItem); } setAddEditItems(testItems[0]); sinon.stub(Zotero.Integration, 'getApplication').callsFake(function(agent, command, docID) { if (!applications[docID]) { applications[docID] = new DocumentPluginDummy.Application(); } return applications[docID]; }); displayDialogStub = sinon.stub(Zotero.Integration, 'displayDialog', // @TODO: This sinon.stub syntax is deprecated! // However when using callsFake tests won't work. This is due to a // possible bug that reset() erases callsFake. // @NOTE: https://github.com/sinonjs/sinon/issues/1341 // displayDialogStub.callsFake(function(doc, dialogName, prefs, io) { function(dialogName, prefs, io) { Zotero.debug(`Display dialog: ${dialogName}`, 2); var ioResult = dialogResults[dialogName.substring(dialogName.lastIndexOf('/')+1, dialogName.length-4)]; if (typeof ioResult == 'function') { ioResult(dialogName, io); } else { Object.assign(io, ioResult); } return Zotero.Promise.resolve(); }); addEditCitationSpy = sinon.spy(Zotero.Integration.Interface.prototype, 'addEditCitation'); }); after(function() { Zotero.Integration.getApplication.restore(); displayDialogStub.restore(); addEditCitationSpy.restore(); }); describe('Interface', function() { describe('#execCommand', function() { var setDocumentDataSpy; var docID = this.fullTitle(); before(function() { setDocumentDataSpy = sinon.spy(DocumentPluginDummy.Document.prototype, 'setDocumentData'); }); afterEach(function() { setDocumentDataSpy.reset(); }); after(function() { setDocumentDataSpy.restore(); }); it('should call doc.setDocumentData once', function* () { yield execCommand('addEditCitation', docID); assert.isTrue(setDocumentDataSpy.calledOnce); }); describe('when style used in the document does not exist', function() { var docID = this.fullTitle(); var displayAlertStub; var style; before(function* () { displayAlertStub = sinon.stub(DocumentPluginDummy.Document.prototype, 'displayAlert').returns(0); }); beforeEach(function() { // 🦉birds? style = {styleID: "http://www.example.com/csl/waterbirds", locale: 'en-US'}; // Make sure style not in library try { Zotero.Styles.get(style.styleID).remove(); } catch (e) {} initDoc(docID, {style}); displayDialogStub.reset(); displayAlertStub.reset(); }); after(function* () { displayAlertStub.restore(); }); describe('when the style is not from a trusted source', function() { it('should download the style and if user clicks YES', function* () { var styleInstallStub = sinon.stub(Zotero.Styles, "install").resolves(); var style = Zotero.Styles.get(styleID); var styleGetCalledOnce = false; var styleGetStub = sinon.stub(Zotero.Styles, 'get').callsFake(function() { if (!styleGetCalledOnce) { styleGetCalledOnce = true; return false; } return style; }); displayAlertStub.returns(1); yield execCommand('addEditCitation', docID); assert.isTrue(displayAlertStub.calledOnce); assert.isFalse(displayDialogStub.calledWith(applications[docID].doc, 'chrome://zotero/content/integration/integrationDocPrefs.xul')); assert.isTrue(styleInstallStub.calledOnce); assert.isOk(Zotero.Styles.get(style.styleID)); styleInstallStub.restore(); styleGetStub.restore(); }); it('should prompt with the document preferences dialog if user clicks NO', function* () { displayAlertStub.returns(0); yield execCommand('addEditCitation', docID); assert.isTrue(displayAlertStub.calledOnce); // Prefs to select a new style and quickFormat assert.isTrue(displayDialogStub.calledTwice); assert.isNotOk(Zotero.Styles.get(style.styleID)); }); }); it('should download the style without prompting if it is from zotero.org', function* (){ initDoc(docID, {styleID: "http://www.zotero.org/styles/waterbirds", locale: 'en-US'}); var styleInstallStub = sinon.stub(Zotero.Styles, "install").resolves(); var style = Zotero.Styles.get(styleID); var styleGetCalledOnce = false; var styleGetStub = sinon.stub(Zotero.Styles, 'get').callsFake(function() { if (!styleGetCalledOnce) { styleGetCalledOnce = true; return false; } return style; }); displayAlertStub.returns(1); yield execCommand('addEditCitation', docID); assert.isFalse(displayAlertStub.called); assert.isFalse(displayDialogStub.calledWith(applications[docID].doc, 'chrome://zotero/content/integration/integrationDocPrefs.xul')); assert.isTrue(styleInstallStub.calledOnce); assert.isOk(Zotero.Styles.get(style.styleID)); styleInstallStub.restore(); styleGetStub.restore(); }); }); }); describe('#addEditCitation', function() { var insertMultipleCitations = Zotero.Promise.coroutine(function *() { var docID = this.test.fullTitle(); if (!(docID in applications)) initDoc(docID); var doc = applications[docID].doc; setAddEditItems(testItems[0]); yield execCommand('addEditCitation', docID); assert.equal(doc.fields.length, 1); var citation = (new Zotero.Integration.CitationField(doc.fields[0], doc.fields[0].code)).unserialize(); assert.equal(citation.citationItems.length, 1); assert.equal(citation.citationItems[0].id, testItems[0].id); setAddEditItems(testItems.slice(1, 3)); yield execCommand('addEditCitation', docID); assert.equal(doc.fields.length, 2); citation = (new Zotero.Integration.CitationField(doc.fields[1], doc.fields[1].code)).unserialize(); assert.equal(citation.citationItems.length, 2); for (let i = 1; i < 3; i++) { assert.equal(citation.citationItems[i-1].id, testItems[i].id); } }); it('should insert citation if not in field', insertMultipleCitations); it('should edit citation if in citation field', function* () { yield insertMultipleCitations.call(this); var docID = this.test.fullTitle(); var doc = applications[docID].doc; sinon.stub(doc, 'cursorInField').returns(doc.fields[0]); sinon.stub(doc, 'canInsertField').returns(false); setAddEditItems(testItems.slice(3, 5)); yield execCommand('addEditCitation', docID); assert.equal(doc.fields.length, 2); var citation = (new Zotero.Integration.CitationField(doc.fields[0], doc.fields[0].code)).unserialize(); assert.equal(citation.citationItems.length, 2); assert.equal(citation.citationItems[0].id, testItems[3].id); }); it('should update bibliography if present', function* () { yield insertMultipleCitations.call(this); var docID = this.test.fullTitle(); var doc = applications[docID].doc; let getCiteprocBibliographySpy = sinon.spy(Zotero.Integration.Bibliography.prototype, 'getCiteprocBibliography'); yield execCommand('addEditBibliography', docID); assert.isTrue(getCiteprocBibliographySpy.calledOnce); assert.equal(getCiteprocBibliographySpy.lastCall.returnValue[0].entry_ids.length, 3); getCiteprocBibliographySpy.reset(); setAddEditItems(testItems[3]); yield execCommand('addEditCitation', docID); assert.equal(getCiteprocBibliographySpy.lastCall.returnValue[0].entry_ids.length, 4); getCiteprocBibliographySpy.restore(); }); describe('when original citation text has been modified', function() { var displayAlertStub; before(function* () { displayAlertStub = sinon.stub(DocumentPluginDummy.Document.prototype, 'displayAlert').returns(0); }); beforeEach(function() { displayAlertStub.reset(); }); after(function() { displayAlertStub.restore(); }); it('should keep modification if "Cancel" selected in editCitation triggered alert', async function () { await insertMultipleCitations.call(this); var docID = this.test.fullTitle(); var doc = applications[docID].doc; doc.fields[0].text = "modified"; sinon.stub(doc, 'cursorInField').returns(doc.fields[0]); sinon.stub(doc, 'canInsertField').returns(false); await execCommand('addEditCitation', docID); assert.equal(doc.fields.length, 2); assert.equal(doc.fields[0].text, "modified"); }); it('should display citation dialog if "OK" selected in editCitation triggered alert', async function () { await insertMultipleCitations.call(this); var docID = this.test.fullTitle(); var doc = applications[docID].doc; let origText = doc.fields[0].text; doc.fields[0].text = "modified"; // Return OK displayAlertStub.returns(1); sinon.stub(doc, 'cursorInField').returns(doc.fields[0]); sinon.stub(doc, 'canInsertField').returns(false); setAddEditItems(testItems[0]); await execCommand('addEditCitation', docID); assert.isTrue(displayAlertStub.called); assert.equal(doc.fields.length, 2); assert.equal(doc.fields[0].text, origText); }); it('should set dontUpdate: true if "yes" selected in refresh prompt', async function() { await insertMultipleCitations.call(this); var docID = this.test.fullTitle(); var doc = applications[docID].doc; var citation = (new Zotero.Integration.CitationField(doc.fields[0], doc.fields[0].code)).unserialize(); assert.isNotOk(citation.properties.dontUpdate); doc.fields[0].text = "modified"; // Return Yes displayAlertStub.returns(1); await execCommand('refresh', docID); assert.isTrue(displayAlertStub.called); assert.equal(doc.fields.length, 2); assert.equal(doc.fields[0].text, "modified"); var citation = (new Zotero.Integration.CitationField(doc.fields[0], doc.fields[0].code)).unserialize(); assert.isOk(citation.properties.dontUpdate); }); it('should reset citation text if "no" selected in refresh prompt', async function() { await insertMultipleCitations.call(this); var docID = this.test.fullTitle(); var doc = applications[docID].doc; var citation = (new Zotero.Integration.CitationField(doc.fields[0], doc.fields[0].code)).unserialize(); assert.isNotOk(citation.properties.dontUpdate); let origText = doc.fields[0].text; doc.fields[0].text = "modified"; // Return No displayAlertStub.returns(0); await execCommand('refresh', docID); assert.isTrue(displayAlertStub.called); assert.equal(doc.fields.length, 2); assert.equal(doc.fields[0].text, origText); var citation = (new Zotero.Integration.CitationField(doc.fields[0], doc.fields[0].code)).unserialize(); assert.isNotOk(citation.properties.dontUpdate); }); }); describe('when delayCitationUpdates is set', function() { it('should insert a citation with wave underlining', function* (){ yield insertMultipleCitations.call(this); var docID = this.test.fullTitle(); var doc = applications[docID].doc; var data = new Zotero.Integration.DocumentData(doc.data); data.prefs.delayCitationUpdates = true; doc.data = data.serialize(); var setTextSpy = sinon.spy(DocumentPluginDummy.Field.prototype, 'setText'); setAddEditItems(testItems[3]); yield execCommand('addEditCitation', docID); assert.isTrue(setTextSpy.lastCall.args[0].includes('\\uldash')); setTextSpy.restore(); }); it('should not write to any other fields besides the one being updated', function* () { yield insertMultipleCitations.call(this); var docID = this.test.fullTitle(); var doc = applications[docID].doc; var data = new Zotero.Integration.DocumentData(doc.data); data.prefs.delayCitationUpdates = true; doc.data = data.serialize(); var setTextSpy = sinon.spy(DocumentPluginDummy.Field.prototype, 'setText'); var setCodeSpy = sinon.spy(DocumentPluginDummy.Field.prototype, 'setCode'); setAddEditItems(testItems[3]); yield execCommand('addEditCitation', docID); var field = setTextSpy.firstCall.thisValue; for (let i = 0; i < setTextSpy.callCount; i++) { assert.isTrue(field.equals(setTextSpy.getCall(i).thisValue)); } for (let i = 0; i < setCodeSpy.callCount; i++) { assert.isTrue(field.equals(setCodeSpy.getCall(i).thisValue)); } setTextSpy.restore(); setCodeSpy.restore(); }) }); }); describe('#addEditBibliography', function() { var docID = this.fullTitle(); beforeEach(function* () { initDoc(docID); yield execCommand('addEditCitation', docID); }); it('should insert bibliography if no bibliography field present', function* () { displayDialogStub.reset(); yield execCommand('addEditBibliography', docID); assert.isFalse(displayDialogStub.called); var biblPresent = false; for (let i = applications[docID].doc.fields.length-1; i >= 0; i--) { let field = Zotero.Integration.Field.loadExisting(applications[docID].doc.fields[i]); if (field.type == INTEGRATION_TYPE_BIBLIOGRAPHY) { biblPresent = true; break; } } assert.isTrue(biblPresent); }); it('should display the edit bibliography dialog if bibliography present', function* () { yield execCommand('addEditBibliography', docID); displayDialogStub.reset(); yield execCommand('addEditBibliography', docID); assert.isTrue(displayDialogStub.calledOnce); assert.isTrue(displayDialogStub.lastCall.args[0].includes('editBibliographyDialog')); }); }); }); describe("DocumentData", function() { it('should properly unserialize old XML document data', function() { var serializedXMLData = "