2015-04-26 21:42:29 +00:00
|
|
|
describe("Zotero.Attachments", function() {
|
2023-04-14 15:37:07 +00:00
|
|
|
var HiddenBrowser;
|
|
|
|
var browser;
|
2015-04-26 21:42:29 +00:00
|
|
|
|
2023-04-14 15:37:07 +00:00
|
|
|
before(function () {
|
|
|
|
HiddenBrowser = ChromeUtils.import("chrome://zotero/content/HiddenBrowser.jsm").HiddenBrowser;
|
2015-04-26 21:42:29 +00:00
|
|
|
});
|
2023-04-14 15:37:07 +00:00
|
|
|
|
|
|
|
afterEach(function () {
|
|
|
|
if (browser) {
|
2023-12-27 09:43:50 +00:00
|
|
|
browser.destroy();
|
2023-04-14 15:37:07 +00:00
|
|
|
browser = null;
|
2015-04-26 21:42:29 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("#importFromFile()", function () {
|
|
|
|
it("should create a child attachment from a text file", function* () {
|
|
|
|
// Create test file
|
|
|
|
var contents = "Test";
|
|
|
|
var tmpFile = Zotero.getTempDirectory();
|
|
|
|
tmpFile.append('test.txt');
|
|
|
|
yield Zotero.File.putContentsAsync(tmpFile, contents);
|
|
|
|
|
|
|
|
// Create parent item
|
|
|
|
var item = new Zotero.Item('book');
|
2015-05-10 08:20:47 +00:00
|
|
|
var parentItemID = yield item.saveTx();
|
2015-04-26 21:42:29 +00:00
|
|
|
|
|
|
|
// Create attachment and compare content
|
2015-05-29 09:31:54 +00:00
|
|
|
var item = yield Zotero.Attachments.importFromFile({
|
2015-05-23 08:25:47 +00:00
|
|
|
file: tmpFile,
|
|
|
|
parentItemID: parentItemID
|
|
|
|
});
|
2015-04-26 21:42:29 +00:00
|
|
|
var storedFile = item.getFile();
|
|
|
|
assert.equal((yield Zotero.File.getContentsAsync(storedFile)), contents);
|
|
|
|
|
|
|
|
// Clean up
|
2015-05-29 09:31:54 +00:00
|
|
|
yield Zotero.Items.erase(item.id);
|
2015-04-26 21:42:29 +00:00
|
|
|
});
|
|
|
|
|
2015-05-23 08:25:47 +00:00
|
|
|
it("should create a top-level attachment from a PNG file", function* () {
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
var contents = yield Zotero.File.getBinaryContentsAsync(file);
|
|
|
|
|
|
|
|
// Create attachment and compare content
|
2015-05-29 09:31:54 +00:00
|
|
|
var item = yield Zotero.Attachments.importFromFile({
|
2015-05-23 08:25:47 +00:00
|
|
|
file: file
|
|
|
|
});
|
|
|
|
var storedFile = item.getFile();
|
|
|
|
assert.equal((yield Zotero.File.getBinaryContentsAsync(storedFile)), contents);
|
|
|
|
|
|
|
|
// Clean up
|
2015-05-29 09:31:54 +00:00
|
|
|
yield Zotero.Items.erase(item.id);
|
2015-05-23 08:25:47 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should create a top-level attachment from a PNG file in a collection", function* () {
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
var contents = yield Zotero.File.getBinaryContentsAsync(file);
|
|
|
|
|
|
|
|
var collection = yield createDataObject('collection');
|
|
|
|
|
|
|
|
// Create attachment and compare content
|
2015-05-29 09:31:54 +00:00
|
|
|
var item = yield Zotero.Attachments.importFromFile({
|
2015-05-23 08:25:47 +00:00
|
|
|
file: file,
|
|
|
|
collections: [collection.id]
|
|
|
|
});
|
|
|
|
var storedFile = item.getFile();
|
|
|
|
assert.equal((yield Zotero.File.getBinaryContentsAsync(storedFile)), contents);
|
|
|
|
|
|
|
|
// Clean up
|
2015-05-29 09:31:54 +00:00
|
|
|
yield Zotero.Items.erase(item.id);
|
2015-05-23 08:25:47 +00:00
|
|
|
});
|
|
|
|
|
2015-04-26 21:42:29 +00:00
|
|
|
it("should create a child attachment from a PNG file", function* () {
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
var contents = yield Zotero.File.getBinaryContentsAsync(file);
|
|
|
|
|
|
|
|
// Create parent item
|
|
|
|
var item = new Zotero.Item('book');
|
2015-05-10 08:20:47 +00:00
|
|
|
var parentItemID = yield item.saveTx();
|
2015-04-26 21:42:29 +00:00
|
|
|
|
|
|
|
// Create attachment and compare content
|
2015-05-29 09:31:54 +00:00
|
|
|
var item = yield Zotero.Attachments.importFromFile({
|
2015-05-23 08:25:47 +00:00
|
|
|
file: file,
|
|
|
|
parentItemID: parentItemID
|
|
|
|
});
|
2015-04-26 21:42:29 +00:00
|
|
|
var storedFile = item.getFile();
|
|
|
|
assert.equal((yield Zotero.File.getBinaryContentsAsync(storedFile)), contents);
|
|
|
|
|
|
|
|
// Clean up
|
2015-05-29 09:31:54 +00:00
|
|
|
yield Zotero.Items.erase(item.id);
|
2015-04-26 21:42:29 +00:00
|
|
|
});
|
2024-07-15 03:37:24 +00:00
|
|
|
|
|
|
|
it("should set a top-level item's title to the filename, minus its extension", async function () {
|
|
|
|
let file = getTestDataDirectory();
|
|
|
|
file.append('test.pdf');
|
|
|
|
let attachment = await Zotero.Attachments.importFromFile({
|
|
|
|
file: file,
|
|
|
|
});
|
|
|
|
assert.equal(attachment.getField('title'), 'test');
|
|
|
|
await attachment.eraseTx();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should set a child item's title to the filename, minus its extension", async function () {
|
|
|
|
let file = getTestDataDirectory();
|
|
|
|
file.append('test.pdf');
|
|
|
|
let parent = await createDataObject('item');
|
|
|
|
let attachment = await Zotero.Attachments.importFromFile({
|
|
|
|
file: file,
|
|
|
|
parentItemID: parent.id,
|
|
|
|
});
|
2024-07-27 07:01:43 +00:00
|
|
|
assert.equal(attachment.getField('title'), Zotero.getString('file-type-pdf'));
|
2024-07-15 03:37:24 +00:00
|
|
|
await parent.eraseTx();
|
|
|
|
});
|
2015-04-26 21:42:29 +00:00
|
|
|
})
|
2015-05-29 05:07:23 +00:00
|
|
|
|
2016-03-22 04:40:59 +00:00
|
|
|
describe("#linkFromFile()", function () {
|
2015-05-29 05:07:23 +00:00
|
|
|
it("should link to a file in My Library", function* () {
|
|
|
|
var item = yield createDataObject('item');
|
|
|
|
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
var attachment = yield Zotero.Attachments.linkFromFile({
|
|
|
|
file: file,
|
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal(attachment.getFilePath(), file.path);
|
|
|
|
})
|
|
|
|
|
|
|
|
it.skip("should throw an error for a non-user library", function* () {
|
|
|
|
// Should create a group library for use by all tests
|
|
|
|
})
|
2024-07-15 03:37:24 +00:00
|
|
|
|
|
|
|
it("should set a top-level item's title to the filename, minus its extension", async function () {
|
|
|
|
let file = getTestDataDirectory();
|
|
|
|
file.append('test.pdf');
|
|
|
|
let attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: file,
|
|
|
|
});
|
|
|
|
assert.equal(attachment.getField('title'), 'test');
|
|
|
|
await attachment.eraseTx();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should set a child item's title to the filename, minus its extension", async function () {
|
|
|
|
let file = getTestDataDirectory();
|
|
|
|
file.append('test.pdf');
|
|
|
|
let parent = await createDataObject('item');
|
|
|
|
let attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: file,
|
|
|
|
parentItemID: parent.id,
|
|
|
|
});
|
2024-07-31 05:39:25 +00:00
|
|
|
assert.equal(attachment.getField('title'), 'test');
|
2024-07-15 03:37:24 +00:00
|
|
|
await parent.eraseTx();
|
|
|
|
});
|
2015-05-29 05:07:23 +00:00
|
|
|
})
|
2015-08-09 08:52:14 +00:00
|
|
|
|
2018-06-30 07:11:13 +00:00
|
|
|
|
|
|
|
describe("#linkFromFileWithRelativePath()", function () {
|
|
|
|
afterEach(function () {
|
|
|
|
Zotero.Prefs.clear('baseAttachmentPath');
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should link to a file using a relative path with no base directory set", async function () {
|
|
|
|
Zotero.Prefs.clear('baseAttachmentPath');
|
|
|
|
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var spy = sinon.spy(Zotero.Fulltext, 'indexPDF');
|
|
|
|
var relPath = 'a/b/test.pdf';
|
|
|
|
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFileWithRelativePath({
|
|
|
|
path: relPath,
|
|
|
|
title: 'test.pdf',
|
|
|
|
parentItemID: item.id,
|
|
|
|
contentType: 'application/pdf'
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.ok(spy.notCalled);
|
|
|
|
spy.restore();
|
|
|
|
assert.equal(
|
|
|
|
attachment.attachmentPath,
|
|
|
|
Zotero.Attachments.BASE_PATH_PLACEHOLDER + relPath
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it("should link to a file using a relative path within the base directory", async function () {
|
|
|
|
var baseDir = await getTempDirectory();
|
|
|
|
Zotero.Prefs.set('baseAttachmentPath', baseDir);
|
|
|
|
Zotero.Prefs.set('saveRelativeAttachmentPath', true);
|
|
|
|
|
|
|
|
var subDir = OS.Path.join(baseDir, 'foo');
|
|
|
|
await OS.File.makeDir(subDir);
|
|
|
|
|
|
|
|
var file = OS.Path.join(subDir, 'test.pdf');
|
|
|
|
await OS.File.copy(OS.Path.join(getTestDataDirectory().path, 'test.pdf'), file);
|
|
|
|
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var spy = sinon.spy(Zotero.Fulltext, 'indexPDF');
|
|
|
|
var relPath = 'foo/test.pdf';
|
|
|
|
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFileWithRelativePath({
|
|
|
|
path: relPath,
|
|
|
|
title: 'test.pdf',
|
|
|
|
parentItemID: item.id,
|
|
|
|
contentType: 'application/pdf'
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.ok(spy.called);
|
|
|
|
spy.restore();
|
|
|
|
assert.equal(
|
|
|
|
attachment.attachmentPath,
|
|
|
|
Zotero.Attachments.BASE_PATH_PLACEHOLDER + relPath
|
|
|
|
);
|
|
|
|
|
|
|
|
assert.ok(await attachment.fileExists());
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it("should link to a nonexistent file using a relative path within the base directory", async function () {
|
|
|
|
var baseDir = await getTempDirectory();
|
|
|
|
Zotero.Prefs.set('baseAttachmentPath', baseDir);
|
|
|
|
Zotero.Prefs.set('saveRelativeAttachmentPath', true);
|
|
|
|
|
|
|
|
var subDir = OS.Path.join(baseDir, 'foo');
|
|
|
|
await OS.File.makeDir(subDir);
|
|
|
|
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var spy = sinon.spy(Zotero.Fulltext, 'indexPDF');
|
|
|
|
var relPath = 'foo/test.pdf';
|
|
|
|
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFileWithRelativePath({
|
|
|
|
path: relPath,
|
|
|
|
title: 'test.pdf',
|
|
|
|
parentItemID: item.id,
|
|
|
|
contentType: 'application/pdf'
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.ok(spy.notCalled);
|
|
|
|
spy.restore();
|
|
|
|
assert.equal(
|
|
|
|
attachment.attachmentPath,
|
|
|
|
Zotero.Attachments.BASE_PATH_PLACEHOLDER + relPath
|
|
|
|
);
|
|
|
|
|
|
|
|
assert.isFalse(await attachment.fileExists());
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it("should reject absolute paths", async function () {
|
|
|
|
try {
|
|
|
|
await Zotero.Attachments.linkFromFileWithRelativePath({
|
|
|
|
path: '/a/b/test.pdf',
|
|
|
|
title: 'test.pdf',
|
|
|
|
contentType: 'application/pdf'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.fail();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2016-03-22 04:40:59 +00:00
|
|
|
describe("#importSnapshotFromFile()", function () {
|
|
|
|
it("should import an HTML file", function* () {
|
|
|
|
var item = yield createDataObject('item');
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.html');
|
|
|
|
var attachment = yield Zotero.Attachments.importSnapshotFromFile({
|
|
|
|
title: 'Snapshot',
|
|
|
|
url: 'http://example.com',
|
|
|
|
file,
|
|
|
|
parentItemID: item.id,
|
|
|
|
contentType: 'text/html',
|
|
|
|
charset: 'utf-8'
|
|
|
|
});
|
|
|
|
|
|
|
|
var matches = yield Zotero.Fulltext.findTextInItems([attachment.id], 'test');
|
|
|
|
assert.lengthOf(matches, 1);
|
|
|
|
assert.propertyVal(matches[0], 'id', attachment.id);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should detect charset for an HTML file", function* () {
|
|
|
|
var item = yield createDataObject('item');
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.html');
|
|
|
|
var attachment = yield Zotero.Attachments.importSnapshotFromFile({
|
|
|
|
title: 'Snapshot',
|
|
|
|
url: 'http://example.com',
|
|
|
|
file,
|
|
|
|
parentItemID: item.id,
|
|
|
|
contentType: 'text/html'
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal(attachment.attachmentCharset, 'utf-8');
|
|
|
|
|
|
|
|
var matches = yield Zotero.Fulltext.findTextInItems([attachment.id], 'test');
|
|
|
|
assert.lengthOf(matches, 1);
|
|
|
|
assert.propertyVal(matches[0], 'id', attachment.id);
|
|
|
|
});
|
2018-06-19 00:17:37 +00:00
|
|
|
|
2022-06-17 08:57:34 +00:00
|
|
|
it("should index JavaScript-created text in an HTML file", async function () {
|
2018-06-19 00:17:37 +00:00
|
|
|
var item = await createDataObject('item');
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test-js.html');
|
|
|
|
var attachment = await Zotero.Attachments.importSnapshotFromFile({
|
|
|
|
title: 'Snapshot',
|
|
|
|
url: 'http://example.com',
|
|
|
|
file,
|
|
|
|
parentItemID: item.id,
|
|
|
|
contentType: 'text/html'
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal(attachment.attachmentCharset, 'utf-8');
|
|
|
|
|
|
|
|
var matches = await Zotero.Fulltext.findTextInItems([attachment.id], 'test');
|
2022-06-17 08:57:34 +00:00
|
|
|
assert.lengthOf(matches, 1);
|
|
|
|
assert.propertyVal(matches[0], 'id', attachment.id);
|
2018-06-19 00:17:37 +00:00
|
|
|
});
|
2016-03-22 04:40:59 +00:00
|
|
|
});
|
|
|
|
|
2021-12-02 09:27:33 +00:00
|
|
|
|
|
|
|
describe("#importFromURL()", function () {
|
2023-12-27 09:43:50 +00:00
|
|
|
it("should use BrowserDownload for a JS redirect page", async function () {
|
|
|
|
let downloadPDFStub = sinon.stub(Zotero.BrowserDownload, "downloadPDF");
|
|
|
|
downloadPDFStub.callsFake(async (_url, path) => {
|
|
|
|
await OS.File.copy(OS.Path.join(getTestDataDirectory().path, 'test.pdf'), path);
|
2021-12-02 09:27:33 +00:00
|
|
|
});
|
2023-12-27 09:43:50 +00:00
|
|
|
try {
|
|
|
|
var item = await Zotero.Attachments.importFromURL({
|
|
|
|
libraryID: Zotero.Libraries.userLibraryID,
|
|
|
|
url: 'https://zotero-static.s3.amazonaws.com/test-pdf-redirect.html',
|
|
|
|
contentType: 'application/pdf'
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.isTrue(downloadPDFStub.calledOnce);
|
|
|
|
}
|
|
|
|
finally {
|
|
|
|
// Clean up
|
|
|
|
await Zotero.Items.erase(item.id);
|
|
|
|
downloadPDFStub.restore();
|
|
|
|
}
|
2021-12-02 09:27:33 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2016-02-11 07:54:52 +00:00
|
|
|
describe("#linkFromDocument", function () {
|
|
|
|
it("should add a link attachment for the current webpage", function* () {
|
|
|
|
var item = yield createDataObject('item');
|
|
|
|
|
|
|
|
var uri = OS.Path.join(getTestDataDirectory().path, "snapshot", "index.html");
|
2023-12-27 09:43:50 +00:00
|
|
|
browser = new HiddenBrowser(uri);
|
|
|
|
yield browser.load(uri);
|
2016-02-11 07:54:52 +00:00
|
|
|
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
var attachment = yield Zotero.Attachments.linkFromDocument({
|
2023-12-27 09:43:50 +00:00
|
|
|
document: yield browser.getDocument(),
|
2016-02-11 07:54:52 +00:00
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal(attachment.getField('url'), "file://" + uri);
|
|
|
|
|
|
|
|
// Check indexing
|
2016-04-09 22:34:54 +00:00
|
|
|
var matches = yield Zotero.Fulltext.findTextInItems([attachment.id], 'share your research');
|
2016-02-11 07:54:52 +00:00
|
|
|
assert.lengthOf(matches, 1);
|
|
|
|
assert.propertyVal(matches[0], 'id', attachment.id);
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2016-05-31 22:03:40 +00:00
|
|
|
describe("#importFromDocument()", function () {
|
2020-07-17 22:14:10 +00:00
|
|
|
Components.utils.import("resource://gre/modules/FileUtils.jsm");
|
2023-08-16 05:10:56 +00:00
|
|
|
|
2020-10-12 23:53:48 +00:00
|
|
|
var testServerPath, httpd, prefix;
|
2023-08-16 05:10:56 +00:00
|
|
|
var testServerPort;
|
2020-07-17 22:14:10 +00:00
|
|
|
|
|
|
|
before(async function () {
|
|
|
|
this.timeout(20000);
|
|
|
|
Zotero.Prefs.set("httpServer.enabled", true);
|
|
|
|
});
|
|
|
|
|
2023-08-16 05:10:56 +00:00
|
|
|
beforeEach(async function () {
|
2020-10-12 23:53:48 +00:00
|
|
|
// Use random prefix because httpd does not actually stop between tests
|
2023-06-19 10:53:01 +00:00
|
|
|
prefix = Zotero.Utilities.randomString();
|
2023-08-16 05:10:56 +00:00
|
|
|
({ httpd, port: testServerPort } = await startHTTPServer());
|
2020-10-12 23:53:48 +00:00
|
|
|
testServerPath = 'http://127.0.0.1:' + testServerPort + '/' + prefix;
|
2020-07-17 22:14:10 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(async function () {
|
|
|
|
var defer = new Zotero.Promise.defer();
|
|
|
|
httpd.stop(() => defer.resolve());
|
|
|
|
await defer.promise;
|
|
|
|
});
|
|
|
|
|
2020-10-23 23:39:07 +00:00
|
|
|
it("should save a document with embedded files", async function () {
|
|
|
|
var item = await createDataObject('item');
|
2020-07-17 22:14:10 +00:00
|
|
|
|
|
|
|
var uri = OS.Path.join(getTestDataDirectory().path, "snapshot");
|
2020-10-12 23:53:48 +00:00
|
|
|
httpd.registerDirectory("/" + prefix + "/", new FileUtils.File(uri));
|
2016-05-31 22:03:40 +00:00
|
|
|
|
2023-12-27 09:43:50 +00:00
|
|
|
browser = new HiddenBrowser();
|
|
|
|
await browser.load(testServerPath + "/index.html");
|
2023-04-15 04:19:27 +00:00
|
|
|
Zotero.FullText.indexNextInTest();
|
2020-10-23 23:39:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.importFromDocument({
|
2023-04-14 15:37:07 +00:00
|
|
|
browser,
|
2016-05-31 22:03:40 +00:00
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
|
2020-07-17 22:14:10 +00:00
|
|
|
assert.equal(attachment.getField('url'), testServerPath + "/index.html");
|
2016-05-31 22:03:40 +00:00
|
|
|
|
|
|
|
// Check indexing
|
2020-10-23 23:39:07 +00:00
|
|
|
var matches = await Zotero.Fulltext.findTextInItems([attachment.id], 'share your research');
|
2016-05-31 22:03:40 +00:00
|
|
|
assert.lengthOf(matches, 1);
|
|
|
|
assert.propertyVal(matches[0], 'id', attachment.id);
|
|
|
|
|
|
|
|
var storageDir = Zotero.Attachments.getStorageDirectory(attachment).path;
|
2020-10-23 23:39:07 +00:00
|
|
|
var file = await attachment.getFilePathAsync();
|
2016-05-31 22:03:40 +00:00
|
|
|
assert.equal(OS.Path.basename(file), 'index.html');
|
2020-07-17 22:14:10 +00:00
|
|
|
|
|
|
|
// Check attachment html file contents
|
|
|
|
let path = OS.Path.join(storageDir, 'index.html');
|
2020-10-23 23:39:07 +00:00
|
|
|
assert.isTrue(await OS.File.exists(path));
|
|
|
|
let contents = await Zotero.File.getContentsAsync(path);
|
2021-01-18 08:52:00 +00:00
|
|
|
assert.include(contents, "><!--\n Page saved with SingleFile");
|
2020-10-23 23:39:07 +00:00
|
|
|
|
|
|
|
// Check attachment base64 contents
|
2020-07-17 22:14:10 +00:00
|
|
|
let expectedPath = getTestDataDirectory();
|
|
|
|
expectedPath.append('snapshot');
|
|
|
|
expectedPath.append('img.gif');
|
2020-10-23 23:39:07 +00:00
|
|
|
let needle = await Zotero.File.getBinaryContentsAsync(expectedPath);
|
|
|
|
needle = '<img src=data:image/gif;base64,' + btoa(needle) + '>';
|
|
|
|
assert.include(contents, needle);
|
2020-07-17 22:14:10 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should save a document with embedded files restricted by CORS", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
|
|
|
|
var url = "file://" + OS.Path.join(getTestDataDirectory().path, "snapshot", "img.gif");
|
|
|
|
httpd.registerPathHandler(
|
2020-10-12 23:53:48 +00:00
|
|
|
'/' + prefix + '/index.html',
|
2020-07-17 22:14:10 +00:00
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 200, "OK");
|
|
|
|
response.write(`<html><head><title>Test</title></head><body><img src="${url}"/>`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2023-12-27 09:43:50 +00:00
|
|
|
let browser = new HiddenBrowser();
|
|
|
|
await browser.load(testServerPath + "/index.html");
|
2020-07-17 22:14:10 +00:00
|
|
|
var attachment = await Zotero.Attachments.importFromDocument({
|
2023-04-14 15:37:07 +00:00
|
|
|
browser,
|
2020-07-17 22:14:10 +00:00
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal(attachment.getField('url'), testServerPath + "/index.html");
|
|
|
|
|
|
|
|
// Check for embedded files
|
|
|
|
var storageDir = Zotero.Attachments.getStorageDirectory(attachment).path;
|
|
|
|
var file = await attachment.getFilePathAsync();
|
|
|
|
assert.equal(OS.Path.basename(file), 'index.html');
|
|
|
|
|
|
|
|
// Check attachment html file contents
|
|
|
|
let path = OS.Path.join(storageDir, 'index.html');
|
|
|
|
assert.isTrue(await OS.File.exists(path));
|
|
|
|
let contents = await Zotero.File.getContentsAsync(path);
|
2021-01-18 08:52:00 +00:00
|
|
|
assert.include(contents, "><!--\n Page saved with SingleFile");
|
2020-07-17 22:14:10 +00:00
|
|
|
|
2020-10-23 23:39:07 +00:00
|
|
|
// Check attachment base64 contents
|
2020-07-17 22:14:10 +00:00
|
|
|
let expectedPath = getTestDataDirectory();
|
|
|
|
expectedPath.append('snapshot');
|
|
|
|
expectedPath.append('img.gif');
|
2020-10-23 23:39:07 +00:00
|
|
|
// This is broken because the browser will not load the image due to CORS and
|
|
|
|
// then SingleFile detects that it is an empty image and replaces it without
|
|
|
|
// trying to load the file. I don't really know of a good way around this for
|
|
|
|
// the moment so I am leaving this assertion commented out, but without the
|
|
|
|
// test is much less useful.
|
|
|
|
// let needle = await Zotero.File.getBinaryContentsAsync(expectedPath);
|
|
|
|
// needle = '<img src=data:image/gif;base64,' + btoa(needle) + '>';
|
2020-10-27 21:58:44 +00:00
|
|
|
// assert.include(contents, needle);
|
2020-07-17 22:14:10 +00:00
|
|
|
});
|
2020-09-28 17:43:32 +00:00
|
|
|
|
|
|
|
it("should save a document with embedded files that throw errors", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
|
|
|
|
var url = "file://" + OS.Path.join(getTestDataDirectory().path, "snapshot", "foobar.gif");
|
|
|
|
httpd.registerPathHandler(
|
2020-10-12 23:53:48 +00:00
|
|
|
'/' + prefix + '/index.html',
|
2020-09-28 17:43:32 +00:00
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 200, "OK");
|
|
|
|
response.write(`<html><head><title>Test</title></head><body><img src="${url}"/>`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2023-12-27 09:43:50 +00:00
|
|
|
let browser = new HiddenBrowser();
|
|
|
|
await browser.load(testServerPath + "/index.html");
|
2020-09-28 17:43:32 +00:00
|
|
|
var attachment = await Zotero.Attachments.importFromDocument({
|
2023-04-14 15:37:07 +00:00
|
|
|
browser,
|
2020-09-28 17:43:32 +00:00
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal(attachment.getField('url'), testServerPath + "/index.html");
|
|
|
|
|
|
|
|
// Check for embedded files
|
|
|
|
var storageDir = Zotero.Attachments.getStorageDirectory(attachment).path;
|
|
|
|
var file = await attachment.getFilePathAsync();
|
|
|
|
assert.equal(OS.Path.basename(file), 'index.html');
|
|
|
|
assert.isFalse(await OS.File.exists(OS.Path.join(storageDir, 'images', '1.gif')));
|
|
|
|
|
|
|
|
// Check attachment html file contents
|
|
|
|
let path = OS.Path.join(storageDir, 'index.html');
|
|
|
|
assert.isTrue(await OS.File.exists(path));
|
|
|
|
let contents = await Zotero.File.getContentsAsync(path);
|
2021-01-18 08:52:00 +00:00
|
|
|
assert.include(contents, "><!--\n Page saved with SingleFile");
|
2020-10-27 21:58:44 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should save a document but not save the iframe", async function () {
|
|
|
|
let item = await createDataObject('item');
|
|
|
|
|
|
|
|
let content = `<html><head><title>Test</title></head><body><iframe src="${testServerPath + "/iframe.html"}"/>`;
|
|
|
|
httpd.registerPathHandler(
|
|
|
|
'/' + prefix + '/index.html',
|
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 200, "OK");
|
|
|
|
response.write(content);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
let url = "file://" + OS.Path.join(getTestDataDirectory().path, "snapshot", "img.gif");
|
|
|
|
httpd.registerPathHandler(
|
|
|
|
'/' + prefix + '/iframe.html',
|
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 200, "OK");
|
|
|
|
response.write(`<html><head><title>Test</title></head><body><img src="${url}"/>`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2023-12-27 09:43:50 +00:00
|
|
|
let browser = new HiddenBrowser();
|
|
|
|
await browser.load(testServerPath + "/index.html");
|
2020-10-27 21:58:44 +00:00
|
|
|
let attachment = await Zotero.Attachments.importFromDocument({
|
2023-04-14 15:37:07 +00:00
|
|
|
browser,
|
2020-10-27 21:58:44 +00:00
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal(attachment.getField('url'), testServerPath + "/index.html");
|
|
|
|
|
|
|
|
// Check for embedded files
|
|
|
|
var storageDir = Zotero.Attachments.getStorageDirectory(attachment).path;
|
|
|
|
var file = await attachment.getFilePathAsync();
|
|
|
|
assert.equal(OS.Path.basename(file), 'index.html');
|
|
|
|
assert.isFalse(await OS.File.exists(OS.Path.join(storageDir, 'images', '1.gif')));
|
|
|
|
|
|
|
|
// Check attachment html file contents
|
|
|
|
let path = OS.Path.join(storageDir, 'index.html');
|
|
|
|
assert.isTrue(await OS.File.exists(path));
|
|
|
|
let contents = await Zotero.File.getContentsAsync(path);
|
|
|
|
assert.include(contents, "><!--\n Page saved with SingleFile");
|
|
|
|
assert.notInclude(contents, "<img src=\"\">'></iframe>");
|
2020-09-28 17:43:32 +00:00
|
|
|
});
|
2020-07-17 22:14:10 +00:00
|
|
|
});
|
|
|
|
|
2020-10-23 23:39:07 +00:00
|
|
|
describe("#importFromSnapshotContent()", function () {
|
|
|
|
it("should save simple HTML content", async function () {
|
2020-07-17 22:14:10 +00:00
|
|
|
let item = await createDataObject('item');
|
|
|
|
|
|
|
|
let content = getTestDataDirectory();
|
|
|
|
content.append('snapshot');
|
|
|
|
content.append('index.html');
|
|
|
|
|
2020-10-23 23:39:07 +00:00
|
|
|
let snapshotContent = await Zotero.File.getContentsAsync(content);
|
2020-07-17 22:14:10 +00:00
|
|
|
|
2023-04-15 04:19:27 +00:00
|
|
|
Zotero.FullText.indexNextInTest();
|
2020-10-23 23:39:07 +00:00
|
|
|
let attachment = await Zotero.Attachments.importFromSnapshotContent({
|
2020-07-17 22:14:10 +00:00
|
|
|
parentItemID: item.id,
|
|
|
|
url: "https://example.com/test.html",
|
|
|
|
title: "Testing Title",
|
2020-10-23 23:39:07 +00:00
|
|
|
snapshotContent
|
2020-07-17 22:14:10 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal(attachment.getField('url'), "https://example.com/test.html");
|
|
|
|
|
|
|
|
// Check indexing
|
|
|
|
let matches = await Zotero.Fulltext.findTextInItems([attachment.id], 'share your research');
|
|
|
|
assert.lengthOf(matches, 1);
|
|
|
|
assert.propertyVal(matches[0], 'id', attachment.id);
|
|
|
|
|
|
|
|
// Check for embedded files
|
|
|
|
let storageDir = Zotero.Attachments.getStorageDirectory(attachment).path;
|
|
|
|
let file = await attachment.getFilePathAsync();
|
|
|
|
assert.equal(OS.Path.basename(file), 'test.html');
|
|
|
|
|
|
|
|
// Check attachment html file contents
|
|
|
|
let path = OS.Path.join(storageDir, 'test.html');
|
|
|
|
assert.isTrue(await OS.File.exists(path));
|
|
|
|
let contents = await Zotero.File.getContentsAsync(path);
|
|
|
|
let expectedContents = await Zotero.File.getContentsAsync(file);
|
|
|
|
assert.equal(contents, expectedContents);
|
2016-05-31 22:03:40 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
describe("Find Full Text", function () {
|
2018-08-07 08:08:47 +00:00
|
|
|
var doiPrefix = 'https://doi.org/';
|
|
|
|
var doi1 = '10.1111/abcd';
|
|
|
|
var doi2 = '10.2222/bcde';
|
|
|
|
var doi3 = '10.3333/cdef';
|
|
|
|
var doi4 = '10.4444/defg';
|
2018-09-06 20:38:28 +00:00
|
|
|
var doi5 = '10.5555/efgh';
|
2018-09-21 08:20:37 +00:00
|
|
|
var doi6 = '10.6666/fghi';
|
2018-08-07 08:08:47 +00:00
|
|
|
var pageURL1 = 'http://website/article1';
|
|
|
|
var pageURL2 = 'http://website/article2';
|
|
|
|
var pageURL3 = 'http://website/article3';
|
|
|
|
var pageURL4 = 'http://website/article4';
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
var pageURL5 = `http://website/${doi4}`;
|
|
|
|
var pageURL6 = `http://website/${doi4}/json`;
|
2018-09-06 20:38:28 +00:00
|
|
|
var pageURL7 = doiPrefix + doi5;
|
2018-09-21 08:20:37 +00:00
|
|
|
var pageURL8 = 'http://website2/article8';
|
|
|
|
var pageURL9 = 'http://website/article9';
|
2018-11-26 07:35:51 +00:00
|
|
|
var pageURL10 = 'http://website/refresh';
|
2024-07-15 15:47:07 +00:00
|
|
|
var pageURL11 = 'http://website/book';
|
2018-08-07 08:08:47 +00:00
|
|
|
|
|
|
|
var httpd;
|
|
|
|
var port = 16213;
|
|
|
|
var baseURL = `http://localhost:${port}/`;
|
2018-09-06 20:38:28 +00:00
|
|
|
var pdfPath = OS.Path.join(getTestDataDirectory().path, 'test.pdf');
|
2018-08-07 08:08:47 +00:00
|
|
|
var pdfURL = `${baseURL}article1/pdf`;
|
|
|
|
var pdfSize;
|
2024-07-15 15:47:07 +00:00
|
|
|
var epubPath = OS.Path.join(getTestDataDirectory().path, 'stub.epub');
|
|
|
|
var epubURL = `${baseURL}article11/epub`;
|
|
|
|
var epubSize;
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
var requestStub;
|
2018-09-21 08:20:37 +00:00
|
|
|
var requestStubCallTimes = [];
|
|
|
|
var return429 = true;
|
2018-08-07 08:08:47 +00:00
|
|
|
|
2018-09-06 20:38:28 +00:00
|
|
|
function makeGetResponseHeader(headers) {
|
|
|
|
return function (header) {
|
|
|
|
if (headers[header] !== undefined) {
|
|
|
|
return headers[header];
|
|
|
|
}
|
2018-09-21 08:20:37 +00:00
|
|
|
throw new Error(`Unimplemented header '${header}'`);
|
2018-09-06 20:38:28 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
function getHTMLPage(includePDF) {
|
|
|
|
return `<html>
|
|
|
|
<head>
|
|
|
|
<title>Page Title</title>
|
|
|
|
<link rel="schema.DC" href="http://purl.org/dc/elements/1.1/" />
|
|
|
|
<meta name="citation_title" content="Title"/>
|
|
|
|
<meta name="${includePDF ? 'citation_pdf_url' : 'ignore'}" content="${pdfURL}"/>
|
|
|
|
</head>
|
|
|
|
<body>Body</body>
|
|
|
|
</html>`;
|
|
|
|
}
|
|
|
|
|
2018-09-06 20:38:28 +00:00
|
|
|
function makeHTMLResponseFromType(html, responseType, responseURL) {
|
|
|
|
var response;
|
|
|
|
if (responseType == 'document') {
|
|
|
|
let parser = new DOMParser();
|
|
|
|
let doc = parser.parseFromString(html, 'text/html');
|
|
|
|
doc = Zotero.HTTP.wrapDocument(doc, responseURL);
|
|
|
|
response = doc;
|
|
|
|
}
|
|
|
|
else if (responseType == 'blob') {
|
|
|
|
let blob = new Blob([html], {type: 'text/html'});
|
|
|
|
response = blob;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
throw new Error("Request not mocked");
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
status: 200,
|
|
|
|
response,
|
|
|
|
responseURL,
|
|
|
|
getResponseHeader: makeGetResponseHeader({
|
|
|
|
'Content-Type': 'text/html'
|
|
|
|
})
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-08-07 08:08:47 +00:00
|
|
|
before(async function () {
|
2018-09-06 20:38:28 +00:00
|
|
|
var pdfBlob = await File.createFromFileName(pdfPath);
|
|
|
|
|
2018-08-07 08:08:47 +00:00
|
|
|
var origFunc = Zotero.HTTP.request.bind(Zotero.HTTP);
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
requestStub = sinon.stub(Zotero.HTTP, 'request');
|
|
|
|
requestStub.callsFake(function (method, url, options) {
|
2018-09-06 20:38:28 +00:00
|
|
|
Zotero.debug("Intercepting " + method + " " + url);
|
2018-09-21 08:20:37 +00:00
|
|
|
requestStubCallTimes.push(new Date());
|
2018-09-06 20:38:28 +00:00
|
|
|
|
2018-08-07 08:08:47 +00:00
|
|
|
// Page responses
|
|
|
|
var routes = [
|
|
|
|
// DOI 1 redirects to page 1, which contains a PDF
|
|
|
|
[doiPrefix + doi1, pageURL1, true],
|
2018-09-21 08:20:37 +00:00
|
|
|
[pageURL1, pageURL1, true],
|
2018-08-07 08:08:47 +00:00
|
|
|
// DOI 2 redirects to page 2, which doesn't contain a PDF, but DOI 2 has an
|
|
|
|
// OA entry for the PDF URL
|
|
|
|
[doiPrefix + doi2, pageURL2, false],
|
2018-09-21 08:20:37 +00:00
|
|
|
[pageURL2, pageURL2, false],
|
2018-08-07 08:08:47 +00:00
|
|
|
// DOI 3 redirects to page 2, which doesn't contain a PDF, but DOI 3 contains
|
|
|
|
// an OA entry for page 3, which contains a PDF)
|
|
|
|
[doiPrefix + doi3, pageURL2, false],
|
|
|
|
[pageURL3, pageURL3, true],
|
|
|
|
// DOI 4 redirects to page 4, which doesn't contain a PDF
|
2018-09-06 20:38:28 +00:00
|
|
|
[doiPrefix + doi4, pageURL4, false],
|
2018-09-21 08:20:37 +00:00
|
|
|
[pageURL4, pageURL4, false],
|
|
|
|
// DOI 6 redirects to page 8, which is on a different domain and has a PDF
|
|
|
|
[doiPrefix + doi6, pageURL8, true],
|
|
|
|
[pageURL8, pageURL8, true],
|
2024-07-15 15:47:07 +00:00
|
|
|
[pageURL11, epubURL, false],
|
2022-10-30 08:44:31 +00:00
|
|
|
|
|
|
|
// Redirect loop
|
|
|
|
['http://website/redirect_loop1', 'http://website/redirect_loop2', false],
|
|
|
|
['http://website/redirect_loop2', 'http://website/redirect_loop3', false],
|
|
|
|
['http://website/redirect_loop3', 'http://website/redirect_loop1', false],
|
|
|
|
|
|
|
|
// Too many total redirects
|
|
|
|
['http://website/too_many_redirects1', 'http://website/too_many_redirects2', false],
|
|
|
|
['http://website/too_many_redirects2', 'http://website/too_many_redirects3', false],
|
|
|
|
['http://website/too_many_redirects3', 'http://website/too_many_redirects4', false],
|
|
|
|
['http://website/too_many_redirects4', 'http://website/too_many_redirects5', false],
|
|
|
|
['http://website/too_many_redirects5', 'http://website/too_many_redirects6', false],
|
|
|
|
['http://website/too_many_redirects6', 'http://website/too_many_redirects7', false],
|
|
|
|
['http://website/too_many_redirects7', 'http://website/too_many_redirects8', false],
|
|
|
|
['http://website/too_many_redirects8', 'http://website/too_many_redirects9', false],
|
|
|
|
['http://website/too_many_redirects9', 'http://website/too_many_redirects10', false],
|
|
|
|
['http://website/too_many_redirects10', 'http://website/too_many_redirects11', false],
|
|
|
|
['http://website/too_many_redirects11', pageURL1, true],
|
2018-08-07 08:08:47 +00:00
|
|
|
];
|
|
|
|
for (let route of routes) {
|
|
|
|
let [expectedURL, responseURL, includePDF] = route;
|
|
|
|
|
|
|
|
if (url != expectedURL) continue;
|
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
// Return explicit 302 if not following redirects
|
|
|
|
if (expectedURL != responseURL && options.followRedirects === false) {
|
|
|
|
return {
|
|
|
|
status: 302,
|
|
|
|
getResponseHeader: makeGetResponseHeader({
|
|
|
|
Location: responseURL
|
|
|
|
})
|
|
|
|
};
|
|
|
|
}
|
2018-09-06 20:38:28 +00:00
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
let html = getHTMLPage(includePDF);
|
2018-09-06 20:38:28 +00:00
|
|
|
return makeHTMLResponseFromType(html, options.responseType, responseURL);
|
2018-08-07 08:08:47 +00:00
|
|
|
}
|
|
|
|
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
// HTML page with PDF download link
|
|
|
|
if (url == pageURL5) {
|
2018-09-06 20:38:28 +00:00
|
|
|
let html = `<html>
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
<head>
|
|
|
|
<title>Page Title</title>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<a id="pdf-link" href="${pdfURL}">Download PDF</a>
|
|
|
|
</body>
|
|
|
|
</html>`;
|
2018-09-06 20:38:28 +00:00
|
|
|
|
|
|
|
return makeHTMLResponseFromType(html, options.responseType, pageURL5);
|
|
|
|
}
|
|
|
|
|
|
|
|
// JSON response with PDF download links
|
|
|
|
if (url == pageURL6) {
|
|
|
|
let response = {
|
|
|
|
oa_locations: [
|
|
|
|
{
|
|
|
|
url_for_landing_page: pageURL1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
url_for_pdf: pdfURL
|
|
|
|
}
|
|
|
|
]
|
|
|
|
};
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
return {
|
|
|
|
status: 200,
|
2018-09-06 20:38:28 +00:00
|
|
|
response,
|
|
|
|
responseURL: pageURL6,
|
|
|
|
getResponseHeader: makeGetResponseHeader({
|
|
|
|
'Content-Type': 'application/json'
|
|
|
|
})
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-09-06 20:38:28 +00:00
|
|
|
// DOI that redirects directly to a PDF
|
|
|
|
if (url == pageURL7) {
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
return {
|
|
|
|
status: 200,
|
2018-09-06 20:38:28 +00:00
|
|
|
response: pdfBlob,
|
|
|
|
responseURL: pdfURL,
|
|
|
|
getResponseHeader: makeGetResponseHeader({
|
|
|
|
'Content-Type': 'application/pdf'
|
|
|
|
})
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
// Returns a 429 every other call
|
|
|
|
if (url.startsWith(pageURL9)) {
|
|
|
|
if (return429) {
|
|
|
|
return429 = false;
|
|
|
|
throw new Zotero.HTTP.UnexpectedStatusException(
|
|
|
|
{
|
|
|
|
status: 429,
|
|
|
|
response: '',
|
|
|
|
responseURL: pageURL9,
|
|
|
|
getResponseHeader: makeGetResponseHeader({
|
|
|
|
'Content-Type': 'text/plain',
|
|
|
|
'Retry-After': '2',
|
|
|
|
})
|
|
|
|
},
|
|
|
|
pageURL9,
|
|
|
|
'Failing with 429'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return429 = true;
|
|
|
|
let html = getHTMLPage(true);
|
|
|
|
return makeHTMLResponseFromType(html, options.responseType, pageURL9);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-26 07:35:51 +00:00
|
|
|
if (url == pageURL10) {
|
|
|
|
let html = `<html><head><meta http-equiv=\"refresh\" content=\"2;url=${pageURL1}\"/></head><body></body></html>`;
|
|
|
|
return makeHTMLResponseFromType(html, options.responseType, pageURL10);
|
|
|
|
}
|
|
|
|
|
2018-08-07 08:08:47 +00:00
|
|
|
// OA PDF lookup
|
|
|
|
if (url.startsWith(ZOTERO_CONFIG.SERVICES_URL)) {
|
|
|
|
let json = JSON.parse(options.body);
|
|
|
|
let response = [];
|
|
|
|
if (json.doi == doi2) {
|
|
|
|
response.push({
|
|
|
|
url: pdfURL,
|
|
|
|
version: 'submittedVersion'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else if (json.doi == doi3) {
|
|
|
|
response.push({
|
|
|
|
pageURL: pageURL3,
|
|
|
|
version: 'submittedVersion'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
status: 200,
|
2018-09-06 20:38:28 +00:00
|
|
|
response,
|
|
|
|
getResponseHeader: makeGetResponseHeader({
|
2022-11-16 23:32:42 +00:00
|
|
|
'Content-Type': 'application/json'
|
2018-09-06 20:38:28 +00:00
|
|
|
})
|
2018-08-07 08:08:47 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
return origFunc(...arguments);
|
|
|
|
});
|
|
|
|
|
2018-09-06 20:38:28 +00:00
|
|
|
pdfSize = await OS.File.stat(pdfPath).size;
|
2024-07-15 15:47:07 +00:00
|
|
|
epubSize = await OS.File.stat(epubPath).size;
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
|
|
|
|
Zotero.Prefs.clear('findPDFs.resolvers');
|
2018-08-07 08:08:47 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
beforeEach(async function () {
|
2023-08-16 05:10:56 +00:00
|
|
|
({ httpd } = await startHTTPServer(port));
|
2018-08-07 08:08:47 +00:00
|
|
|
httpd.registerFile(
|
2024-07-15 15:47:07 +00:00
|
|
|
pdfURL.substring(baseURL.length - 1),
|
|
|
|
Zotero.File.pathToFile(pdfPath)
|
|
|
|
);
|
|
|
|
httpd.registerFile(
|
|
|
|
epubURL.substring(baseURL.length - 1),
|
|
|
|
Zotero.File.pathToFile(epubPath)
|
2018-08-07 08:08:47 +00:00
|
|
|
);
|
2018-09-21 08:20:37 +00:00
|
|
|
|
2022-11-16 23:32:17 +00:00
|
|
|
// Generate a page with a relative PDF URL
|
|
|
|
httpd.registerPathHandler(
|
|
|
|
"/" + doi4,
|
|
|
|
{
|
|
|
|
handle: function (request, response) {
|
|
|
|
response.setStatusLine(null, 200, "OK");
|
|
|
|
response.write(`<html>
|
|
|
|
<head>
|
|
|
|
<title>Page Title</title>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<a id="pdf-link" href="/article1/pdf">Download PDF</a>
|
|
|
|
</body>
|
|
|
|
</html>`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
requestStubCallTimes = [];
|
2018-08-07 08:08:47 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(async function () {
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
requestStub.resetHistory();
|
2018-08-07 08:08:47 +00:00
|
|
|
await new Promise((resolve) => {
|
|
|
|
httpd.stop(() => resolve());
|
|
|
|
});
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
Zotero.Prefs.clear('findPDFs.resolvers');
|
2018-10-06 05:38:32 +00:00
|
|
|
|
|
|
|
// Close progress dialog after each run
|
2024-07-15 15:47:07 +00:00
|
|
|
var queue = Zotero.ProgressQueues.get('findFile');
|
2018-10-06 05:38:32 +00:00
|
|
|
if (queue) {
|
|
|
|
queue.getDialog().close();
|
|
|
|
}
|
2018-08-07 08:08:47 +00:00
|
|
|
}.bind(this));
|
|
|
|
|
|
|
|
after(() => {
|
|
|
|
Zotero.HTTP.request.restore();
|
|
|
|
});
|
|
|
|
|
2018-09-06 20:38:28 +00:00
|
|
|
it("should add a PDF from a resolved DOI webpage", async function () {
|
2018-08-07 08:08:47 +00:00
|
|
|
var doi = doi1;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2018-09-06 20:38:28 +00:00
|
|
|
|
2023-04-29 21:49:32 +00:00
|
|
|
assert.equal(requestStub.callCount, 2);
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.isTrue(requestStub.getCall(0).calledWith('GET', 'https://doi.org/' + doi));
|
2018-09-06 20:38:28 +00:00
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should add a PDF from a DOI that resolves directly to the file", async function () {
|
|
|
|
var doi = doi5;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2018-08-07 08:08:47 +00:00
|
|
|
|
2023-04-29 21:49:32 +00:00
|
|
|
assert.equal(requestStub.callCount, 1);
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
assert.isTrue(requestStub.calledWith('GET', 'https://doi.org/' + doi));
|
2018-08-07 08:08:47 +00:00
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
2018-08-30 19:13:04 +00:00
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should add a PDF from a resolved DOI from the Extra field", async function () {
|
|
|
|
var doi = doi1;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('extra', 'DOI: ' + doi);
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2018-08-30 19:13:04 +00:00
|
|
|
|
2023-04-29 21:49:32 +00:00
|
|
|
assert.equal(requestStub.callCount, 2);
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.isTrue(requestStub.getCall(0).calledWith('GET', 'https://doi.org/' + doi));
|
2018-08-30 19:13:04 +00:00
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
2018-08-07 08:08:47 +00:00
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should add a PDF from a URL", async function () {
|
|
|
|
var url = pageURL1;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('url', url);
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2018-08-07 08:08:47 +00:00
|
|
|
|
2023-04-29 21:49:32 +00:00
|
|
|
assert.equal(requestStub.callCount, 1);
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
assert.isTrue(requestStub.calledWith('GET', url));
|
2018-08-07 08:08:47 +00:00
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
it("should add an EPUB from a URL with a redirect", async function () {
|
|
|
|
var url = pageURL11;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'book' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('url', url);
|
|
|
|
await item.saveTx();
|
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
|
|
|
assert.equal(requestStub.callCount, 2);
|
|
|
|
var call = requestStub.getCall(0);
|
|
|
|
assert.isTrue(call.calledWith('GET', url));
|
|
|
|
call = requestStub.getCall(1);
|
|
|
|
assert.isTrue(call.calledWith('GET', epubURL));
|
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, epubURL);
|
|
|
|
assert.equal(json.contentType, 'application/epub+zip');
|
|
|
|
assert.equal(json.filename, 'Test.epub');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, epubSize);
|
|
|
|
});
|
|
|
|
|
2018-08-07 08:08:47 +00:00
|
|
|
it("should add an OA PDF from a direct URL", async function () {
|
|
|
|
var doi = doi2;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2018-08-07 08:08:47 +00:00
|
|
|
|
2023-04-29 21:49:32 +00:00
|
|
|
assert.equal(requestStub.callCount, 3);
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
var call1 = requestStub.getCall(0);
|
2018-08-07 08:08:47 +00:00
|
|
|
assert.isTrue(call1.calledWith('GET', 'https://doi.org/' + doi));
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
var call2 = requestStub.getCall(1);
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.isTrue(call2.calledWith('GET', pageURL2));
|
|
|
|
var call3 = requestStub.getCall(2);
|
|
|
|
assert.isTrue(call3.calledWith('POST', ZOTERO_CONFIG.SERVICES_URL + 'oa/search'));
|
2018-08-07 08:08:47 +00:00
|
|
|
|
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should add an OA PDF from a page URL", async function () {
|
|
|
|
var doi = doi3;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2018-08-07 08:08:47 +00:00
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.equal(requestStub.callCount, 4);
|
2018-08-07 08:08:47 +00:00
|
|
|
// Check the DOI (and get nothing)
|
2018-09-21 08:20:37 +00:00
|
|
|
var call = requestStub.getCall(0);
|
|
|
|
assert.isTrue(call.calledWith('GET', 'https://doi.org/' + doi));
|
|
|
|
call = requestStub.getCall(1);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL2));
|
2018-08-07 08:08:47 +00:00
|
|
|
// Check the OA resolver and get page 3
|
2018-09-21 08:20:37 +00:00
|
|
|
call = requestStub.getCall(2);
|
|
|
|
assert.isTrue(call.calledWith('POST', ZOTERO_CONFIG.SERVICES_URL + 'oa/search'));
|
2018-08-07 08:08:47 +00:00
|
|
|
// Check page 3 and find the download URL
|
2018-09-21 08:20:37 +00:00
|
|
|
call = requestStub.getCall(3);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL3));
|
2018-08-07 08:08:47 +00:00
|
|
|
|
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
2020-02-03 04:34:57 +00:00
|
|
|
it("shouldn't try the URL-field URL again if it was already checked as the redirected DOI URL", async function () {
|
2018-08-07 08:08:47 +00:00
|
|
|
var doi = doi4;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
item.setField('url', pageURL4);
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2018-08-07 08:08:47 +00:00
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.equal(requestStub.callCount, 3);
|
|
|
|
var call = requestStub.getCall(0);
|
|
|
|
assert.isTrue(call.calledWith('GET', 'https://doi.org/' + doi));
|
|
|
|
call = requestStub.getCall(1);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL4));
|
|
|
|
call = requestStub.getCall(2);
|
|
|
|
assert.isTrue(call.calledWith('POST', ZOTERO_CONFIG.SERVICES_URL + 'oa/search'));
|
2018-08-07 08:08:47 +00:00
|
|
|
|
|
|
|
assert.isFalse(attachment);
|
|
|
|
});
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
it("should wait between requests to the same domain", async function () {
|
|
|
|
var url1 = pageURL1;
|
|
|
|
var item1 = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item1.setField('title', 'Test');
|
|
|
|
item1.setField('url', url1);
|
|
|
|
await item1.saveTx();
|
|
|
|
|
|
|
|
var url2 = pageURL3;
|
|
|
|
var item2 = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item2.setField('title', 'Test');
|
|
|
|
item2.setField('url', url2);
|
|
|
|
await item2.saveTx();
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachments = await Zotero.Attachments.addAvailableFiles([item1, item2]);
|
2018-09-21 08:20:37 +00:00
|
|
|
|
2023-04-29 21:49:32 +00:00
|
|
|
assert.equal(requestStub.callCount, 2);
|
2019-01-29 12:35:39 +00:00
|
|
|
assert.isAbove(requestStubCallTimes[1] - requestStubCallTimes[0], 998);
|
2018-10-06 05:38:32 +00:00
|
|
|
// Make sure both items have attachments
|
|
|
|
assert.equal(item1.numAttachments(), 1);
|
|
|
|
assert.equal(item2.numAttachments(), 1);
|
2018-09-21 08:20:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should wait between requests that resolve to the same domain", async function () {
|
|
|
|
// DOI URL resolves to 'website' domain with PDF
|
|
|
|
var url1 = doiPrefix + doi1;
|
|
|
|
var item1 = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item1.setField('title', 'Test');
|
|
|
|
item1.setField('url', url1);
|
|
|
|
await item1.saveTx();
|
|
|
|
|
|
|
|
// DOI URL resolves to 'website' domain without PDF
|
|
|
|
var url2 = doiPrefix + doi4;
|
|
|
|
var item2 = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item2.setField('title', 'Test');
|
|
|
|
item2.setField('url', url2);
|
|
|
|
await item2.saveTx();
|
|
|
|
|
|
|
|
// DOI URL resolves to 'website2' domain without PDF
|
|
|
|
var url3 = doiPrefix + doi6;
|
|
|
|
var item3 = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item3.setField('title', 'Test');
|
|
|
|
item3.setField('url', url3);
|
|
|
|
await item3.saveTx();
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachments = await Zotero.Attachments.addAvailableFiles([item1, item2, item3]);
|
2018-09-21 08:20:37 +00:00
|
|
|
|
|
|
|
assert.equal(requestStub.callCount, 6);
|
|
|
|
assert.equal(requestStub.getCall(0).args[1], doiPrefix + doi1);
|
|
|
|
assert.equal(requestStub.getCall(1).args[1], pageURL1);
|
|
|
|
assert.equal(requestStub.getCall(2).args[1], doiPrefix + doi4);
|
|
|
|
// Should skip ahead to the next DOI
|
|
|
|
assert.equal(requestStub.getCall(3).args[1], doiPrefix + doi6);
|
|
|
|
// which is on a new domain
|
|
|
|
assert.equal(requestStub.getCall(4).args[1], pageURL8);
|
|
|
|
// and then return to make 'website' request for DOI 4
|
|
|
|
assert.equal(requestStub.getCall(5).args[1], pageURL4);
|
|
|
|
|
|
|
|
// 'website' requests should be a second apart
|
2021-02-09 22:12:25 +00:00
|
|
|
assert.isAbove(requestStubCallTimes[5] - requestStubCallTimes[1], 995);
|
2018-09-21 08:20:37 +00:00
|
|
|
|
2018-10-06 05:38:32 +00:00
|
|
|
assert.equal(item1.numAttachments(), 1);
|
|
|
|
assert.equal(item2.numAttachments(), 0);
|
|
|
|
assert.equal(item3.numAttachments(), 1);
|
2018-09-21 08:20:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should wait between requests to the same domain after a 429", async function () {
|
|
|
|
var url1 = pageURL9;
|
|
|
|
var item1 = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item1.setField('title', 'Test');
|
|
|
|
item1.setField('url', url1);
|
|
|
|
await item1.saveTx();
|
|
|
|
|
|
|
|
var url2 = pageURL3;
|
|
|
|
var item2 = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item2.setField('title', 'Test');
|
|
|
|
item2.setField('url', url2);
|
|
|
|
await item2.saveTx();
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachments = await Zotero.Attachments.addAvailableFiles([item1, item2]);
|
2018-09-21 08:20:37 +00:00
|
|
|
|
2023-04-29 21:49:32 +00:00
|
|
|
assert.equal(requestStub.callCount, 3);
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.equal(requestStub.getCall(0).args[1], pageURL9);
|
|
|
|
assert.equal(requestStub.getCall(1).args[1], pageURL9);
|
|
|
|
assert.equal(requestStub.getCall(2).args[1], pageURL3);
|
2018-10-09 23:14:59 +00:00
|
|
|
assert.isAbove(requestStubCallTimes[1] - requestStubCallTimes[0], 1999);
|
2018-10-06 05:38:32 +00:00
|
|
|
// Make sure both items have attachments
|
|
|
|
assert.equal(item1.numAttachments(), 1);
|
|
|
|
assert.equal(item2.numAttachments(), 1);
|
2018-09-21 08:20:37 +00:00
|
|
|
});
|
|
|
|
|
2018-11-26 07:35:51 +00:00
|
|
|
it("should follow a meta redirect", async function () {
|
|
|
|
var url = pageURL10;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('url', url);
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2018-11-26 07:35:51 +00:00
|
|
|
|
2023-04-29 21:49:32 +00:00
|
|
|
assert.equal(requestStub.callCount, 2);
|
2018-11-26 07:35:51 +00:00
|
|
|
assert.equal(requestStub.getCall(0).args[1], pageURL10)
|
|
|
|
assert.equal(requestStub.getCall(1).args[1], pageURL1)
|
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
2022-10-30 08:44:31 +00:00
|
|
|
it("should stop after too many redirects to the same URL", async function () {
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('url', 'http://website/redirect_loop1');
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2022-10-30 08:44:31 +00:00
|
|
|
assert.isFalse(attachment);
|
|
|
|
assert.equal(requestStub.callCount, 7);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should stop after too many total redirects for a given page URL", async function () {
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('url', 'http://website/too_many_redirects1');
|
|
|
|
await item.saveTx();
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2022-10-30 08:44:31 +00:00
|
|
|
assert.isFalse(attachment);
|
|
|
|
assert.equal(requestStub.callCount, 10);
|
|
|
|
});
|
|
|
|
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
it("should handle a custom resolver in HTML mode", async function () {
|
|
|
|
var doi = doi4;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var resolvers = [{
|
|
|
|
name: 'Custom',
|
|
|
|
method: 'get',
|
|
|
|
url: 'http://website/{doi}',
|
|
|
|
mode: 'html',
|
|
|
|
selector: '#pdf-link',
|
|
|
|
attribute: 'href'
|
|
|
|
}];
|
|
|
|
Zotero.Prefs.set('findPDFs.resolvers', JSON.stringify(resolvers));
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.equal(requestStub.callCount, 4);
|
|
|
|
var call = requestStub.getCall(0);
|
|
|
|
assert.isTrue(call.calledWith('GET', 'https://doi.org/' + doi));
|
|
|
|
var call = requestStub.getCall(1);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL4));
|
|
|
|
call = requestStub.getCall(2);
|
|
|
|
assert.isTrue(call.calledWith('POST', ZOTERO_CONFIG.SERVICES_URL + 'oa/search'));
|
|
|
|
call = requestStub.getCall(3);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL5));
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
|
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
2022-11-16 23:32:17 +00:00
|
|
|
it("should handle a custom resolver with a relative PDF path in HTML mode", async function () {
|
|
|
|
var doi = doi4;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var resolvers = [{
|
|
|
|
name: 'Custom',
|
|
|
|
method: 'get',
|
|
|
|
// Registered with httpd.js in beforeEach()
|
|
|
|
url: baseURL + "{doi}",
|
|
|
|
mode: 'html',
|
|
|
|
selector: '#pdf-link',
|
|
|
|
attribute: 'href'
|
|
|
|
}];
|
|
|
|
Zotero.Prefs.set('findPDFs.resolvers', JSON.stringify(resolvers));
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
2022-11-16 23:32:17 +00:00
|
|
|
|
|
|
|
assert.equal(requestStub.callCount, 4);
|
|
|
|
var call = requestStub.getCall(0);
|
|
|
|
assert.isTrue(call.calledWith('GET', 'https://doi.org/' + doi));
|
|
|
|
var call = requestStub.getCall(1);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL4));
|
|
|
|
call = requestStub.getCall(2);
|
|
|
|
assert.isTrue(call.calledWith('POST', ZOTERO_CONFIG.SERVICES_URL + 'oa/search'));
|
|
|
|
var call = requestStub.getCall(3);
|
|
|
|
assert.isTrue(call.calledWith('GET', baseURL + doi4));
|
|
|
|
|
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
it("should handle a custom resolver in JSON mode with URL strings", async function () {
|
|
|
|
var doi = doi4;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var resolvers = [{
|
|
|
|
name: 'Custom',
|
|
|
|
method: 'get',
|
|
|
|
url: 'http://website/{doi}/json',
|
|
|
|
mode: 'json',
|
|
|
|
selector: '.oa_locations.url_for_pdf'
|
|
|
|
}];
|
|
|
|
Zotero.Prefs.set('findPDFs.resolvers', JSON.stringify(resolvers));
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.equal(requestStub.callCount, 4);
|
|
|
|
var call = requestStub.getCall(0);
|
|
|
|
assert.isTrue(call.calledWith('GET', 'https://doi.org/' + doi));
|
|
|
|
call = requestStub.getCall(1);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL4));
|
|
|
|
call = requestStub.getCall(2);
|
|
|
|
assert.isTrue(call.calledWith('POST', ZOTERO_CONFIG.SERVICES_URL + 'oa/search'));
|
|
|
|
call = requestStub.getCall(3);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL6));
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
|
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle a custom resolver in JSON mode with mapped properties", async function () {
|
|
|
|
var doi = doi4;
|
|
|
|
var item = createUnsavedDataObject('item', { itemType: 'journalArticle' });
|
|
|
|
item.setField('title', 'Test');
|
|
|
|
item.setField('DOI', doi);
|
|
|
|
await item.saveTx();
|
|
|
|
|
|
|
|
var resolvers = [{
|
|
|
|
name: 'Custom',
|
|
|
|
method: 'get',
|
|
|
|
url: 'http://website/{doi}/json',
|
|
|
|
mode: 'json',
|
|
|
|
selector: '.oa_locations',
|
|
|
|
mappings: {
|
|
|
|
url: 'url_for_pdf',
|
|
|
|
pageURL: 'url_for_landing_page',
|
|
|
|
}
|
|
|
|
}];
|
|
|
|
Zotero.Prefs.set('findPDFs.resolvers', JSON.stringify(resolvers));
|
|
|
|
|
2024-07-15 15:47:07 +00:00
|
|
|
var attachment = await Zotero.Attachments.addAvailableFile(item);
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
|
2018-09-21 08:20:37 +00:00
|
|
|
assert.equal(requestStub.callCount, 5);
|
|
|
|
var call = requestStub.getCall(0);
|
|
|
|
assert.isTrue(call.calledWith('GET', 'https://doi.org/' + doi));
|
|
|
|
call = requestStub.getCall(1);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL4));
|
|
|
|
call = requestStub.getCall(2);
|
|
|
|
assert.isTrue(call.calledWith('POST', ZOTERO_CONFIG.SERVICES_URL + 'oa/search'));
|
|
|
|
call = requestStub.getCall(3);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL6));
|
|
|
|
call = requestStub.getCall(4);
|
|
|
|
assert.isTrue(call.calledWith('GET', pageURL1));
|
Automatically download open-access PDFs when saving via the connector
If there's no translated PDF or the translated PDF fails and the item
has a DOI, check Zotero's Unpaywall mirror for possible sources and try
to download one of those.
Unlike with "Add Item by Identifier" and "Find Available PDF" in the
item context menu, this does not try the DOI/URL page, since it would
result in more data leakage and most of the time you'd be saving from
the DOI page already. We could consider offering it as an option, but
for it to be useful, you'd have to have an institutional subscription,
be on-campus or connected via VPN (for now), and be saving from
somewhere other than the main page.
A new connector endpoint, sessionProgress, takes the place of
attachmentProgress. Unlike attachmentProgress, sessionProgress can show
new attachments that have been added to the save, and with a little more
work should also be able to show when a parent item has been recognized
for a directly saved PDF.
This also adds support for custom PDF resolvers, available to all PDF
retrieval methods. I'll document those separately.
Closes #1542
2018-08-15 07:34:28 +00:00
|
|
|
|
|
|
|
assert.ok(attachment);
|
|
|
|
var json = attachment.toJSON();
|
|
|
|
assert.equal(json.url, pdfURL);
|
|
|
|
assert.equal(json.contentType, 'application/pdf');
|
|
|
|
assert.equal(json.filename, 'Test.pdf');
|
|
|
|
assert.equal(await OS.File.stat(attachment.getFilePath()).size, pdfSize);
|
|
|
|
});
|
2018-08-07 08:08:47 +00:00
|
|
|
});
|
|
|
|
|
2020-06-02 19:31:12 +00:00
|
|
|
describe("#getFileBaseNameFromItem()", function () {
|
2024-07-27 15:39:10 +00:00
|
|
|
var item, itemManyAuthors, itemPatent, itemIncomplete, itemBookSection, itemSpaces, itemSuffixes, itemKeepHyphens,
|
|
|
|
itemNoRepeatedHyphens, itemNoRepeatedUnderscores;
|
2023-07-20 10:50:34 +00:00
|
|
|
|
|
|
|
before(() => {
|
|
|
|
item = createUnsavedDataObject('item', { title: 'Lorem Ipsum', itemType: 'journalArticle' });
|
|
|
|
item.setCreators([
|
|
|
|
{ firstName: 'Foocius', lastName: 'Barius', creatorType: 'author' },
|
|
|
|
{ firstName: 'Bazius', lastName: 'Pixelus', creatorType: 'author' }
|
|
|
|
]);
|
|
|
|
item.setField('date', "1975-10-15");
|
|
|
|
item.setField('publicationTitle', 'Best Publications Place');
|
|
|
|
item.setField('journalAbbreviation', 'BPP');
|
|
|
|
item.setField('issue', '42');
|
|
|
|
item.setField('pages', '321');
|
|
|
|
|
2024-07-27 15:39:10 +00:00
|
|
|
itemBookSection = createUnsavedDataObject('item', { title: 'Book Section', itemType: 'bookSection' });
|
|
|
|
itemBookSection.setField('bookTitle', 'Book Title');
|
|
|
|
|
2023-07-20 10:50:34 +00:00
|
|
|
itemManyAuthors = createUnsavedDataObject('item', { title: 'Has Many Authors', itemType: 'book' });
|
|
|
|
itemManyAuthors.setCreators([
|
|
|
|
{ firstName: 'First', lastName: 'Author', creatorType: 'author' },
|
|
|
|
{ firstName: 'Second', lastName: 'Creator', creatorType: 'author' },
|
|
|
|
{ firstName: 'Third', lastName: 'Person', creatorType: 'author' },
|
|
|
|
{ firstName: 'Final', lastName: 'Writer', creatorType: 'author' },
|
|
|
|
{ firstName: 'Some', lastName: 'Editor1', creatorType: 'editor' },
|
|
|
|
{ firstName: 'Other', lastName: 'ProEditor2', creatorType: 'editor' },
|
|
|
|
{ firstName: 'Last', lastName: 'SuperbEditor3', creatorType: 'editor' },
|
|
|
|
]);
|
|
|
|
itemManyAuthors.setField('date', "2000-01-02");
|
|
|
|
itemManyAuthors.setField('publisher', 'Awesome House');
|
|
|
|
itemManyAuthors.setField('volume', '3');
|
|
|
|
|
|
|
|
itemPatent = createUnsavedDataObject('item', { title: 'Retroencabulator', itemType: 'patent' });
|
|
|
|
itemPatent.setCreators([
|
|
|
|
{ name: 'AcmeCorp', creatorType: 'inventor' },
|
|
|
|
{ firstName: 'Wile', lastName: 'E', creatorType: 'contributor' },
|
|
|
|
{ firstName: 'Road', lastName: 'R', creatorType: 'contributor' },
|
|
|
|
]);
|
|
|
|
itemPatent.setField('date', '1952-05-10');
|
|
|
|
itemPatent.setField('number', 'HBK-8539b');
|
|
|
|
itemPatent.setField('assignee', 'Fast FooBar');
|
|
|
|
itemIncomplete = createUnsavedDataObject('item', { title: 'Incomplete', itemType: 'preprint' });
|
2024-07-27 15:39:10 +00:00
|
|
|
|
2024-02-19 09:43:19 +00:00
|
|
|
itemSpaces = createUnsavedDataObject('item', { title: ' Spaces! ', itemType: 'book' });
|
2024-07-16 05:59:07 +00:00
|
|
|
itemSuffixes = createUnsavedDataObject('item', { title: '-Suffixes-', itemType: 'book' });
|
|
|
|
itemSuffixes.setField('date', "1999-07-15");
|
2024-07-27 15:39:10 +00:00
|
|
|
itemKeepHyphens = createUnsavedDataObject('item', { title: 'keep--hyphens', itemType: 'journalArticle' });
|
|
|
|
itemKeepHyphens.setField('publicationTitle', "keep");
|
|
|
|
itemKeepHyphens.setField('issue', 'hyphens');
|
|
|
|
itemKeepHyphens.setField('date', "1999-07-15");
|
|
|
|
itemNoRepeatedHyphens = createUnsavedDataObject('item', { title: 'no - repeated - hyphens', itemType: 'journalArticle' });
|
|
|
|
itemNoRepeatedHyphens.setField('publicationTitle', "no- repeated- hyphens");
|
|
|
|
itemNoRepeatedUnderscores = createUnsavedDataObject('item', { title: 'no _ repeated _ underscores', itemType: 'journalArticle' });
|
|
|
|
itemNoRepeatedUnderscores.setField('publicationTitle', "no_ repeated_ underscores");
|
2023-07-20 10:50:34 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should strip HTML tags from title', function () {
|
|
|
|
var htmlItem = createUnsavedDataObject('item', { title: 'Foo <i>Bar</i> Foo<br><br/><br />Bar' });
|
2024-07-27 15:39:10 +00:00
|
|
|
var str = Zotero.Attachments.getFileBaseNameFromItem(htmlItem, { formatString: '{{ title }}' });
|
2020-06-02 19:31:12 +00:00
|
|
|
assert.equal(str, 'Foo Bar Foo Bar');
|
|
|
|
});
|
2023-07-20 10:50:34 +00:00
|
|
|
|
|
|
|
it('should accept basic formating options', function () {
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: 'FOO{{year}}BAR' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'FOO1975BAR'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{firstCreator suffix=" - "}}{{year suffix=" - "}}{{title truncate="50" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Barius and Pixelus - 1975 - Lorem Ipsum'
|
|
|
|
);
|
2024-01-12 18:24:40 +00:00
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{firstCreator suffix=" - " replaceFrom=" *and *" replaceTo="&"}}{{year suffix=" - " replaceFrom="(\\d{2})(\\d{2})" replaceTo="$2"}}{{title truncate="50" replaceFrom=".m" replaceTo="a"}} - {{title truncate="50" replaceFrom=".m" replaceTo="a" regexOpts="g"}}' }),
|
2024-01-12 18:24:40 +00:00
|
|
|
'Barius&Pixelus - 75 - Lora Ipsum - Lora Ipsa'
|
|
|
|
);
|
2023-07-20 10:50:34 +00:00
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{year suffix="-"}}{{firstCreator truncate="10" suffix="-"}}{{title truncate="5" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'1975-Barius and-Lorem'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: 'foo {{year}} bar {{year prefix="++" truncate="2" suffix="++"}}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'foo 1975 bar ++19++'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{firstCreator suffix=" - "}}{{year suffix=" - "}}{{title}}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Author et al. - 2000 - Has Many Authors'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-02-19 09:43:19 +00:00
|
|
|
it('should trim whitespaces from a value', function () {
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemSpaces, { formatString: '{{ title }}' }),
|
2024-02-19 09:43:19 +00:00
|
|
|
'Spaces!'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{title truncate="6"}}' }),
|
2024-02-19 09:43:19 +00:00
|
|
|
'Lorem'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{firstCreator truncate="7"}}' }),
|
2024-02-19 09:43:19 +00:00
|
|
|
'Barius'
|
|
|
|
);
|
|
|
|
// but preserve if it's configured as a prefix or suffix
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{title prefix=" " suffix=" "}}' }),
|
2024-02-19 09:43:19 +00:00
|
|
|
' Lorem Ipsum '
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-07-27 15:39:10 +00:00
|
|
|
it('should offer a range of options for composing creators', async function () {
|
2023-07-20 10:50:34 +00:00
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ authors max="1" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Barius'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ authors max="1" truncate="3" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Bar'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ authors max="5" join=" " }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Barius Pixelus'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ authors max="3" join=" " }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Author Creator Person'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemPatent, { formatString: '{{ authors }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'AcmeCorp'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ authors max="2" name="family" initialize="family" join=" " initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'A C'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemPatent, { formatString: '{{ authors max="2" name="family" initialize="family" initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'A'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ authors max="1" name="full" initialize="full" name-part-separator="" initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'FB'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ authors max="3" name="full" initialize="full" name-part-separator="" join=" " initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'FA SC TP'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ authors max="1" name="family-given" initialize="given" name-part-separator="" initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'BariusF'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ authors max="2" name="family-given" initialize="given" join=" " name-part-separator="" initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'AuthorF CreatorS'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ editors }}test' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'test'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ editors max="1" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Editor1'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ editors max="5" join=" " }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Editor1 ProEditor2 SuperbEditor3'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ editors max="2" name="family" initialize="family" join=" " initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'E P'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ editors max="1" name="full" initialize="full" name-part-separator="" initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'SE'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ editors max="1" name="family-given" initialize="given" name-part-separator="" initialize-with="" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Editor1S'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ authors max="3" name="full" initialize="given" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'F. Barius, B. Pixelus'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ creators case="upper" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'BARIUS, PIXELUS'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ authors max="2" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Author, Creator'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ creators max="3" join=" " name="given" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'First Second Third'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should accept case parameter', async function () {
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ publicationTitle case="upper" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'BEST PUBLICATIONS PLACE'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ publicationTitle case="lower" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'best publications place'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ publicationTitle case="title" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Best Publications Place'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ publicationTitle case="hyphen" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'best-publications-place'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ publicationTitle case="camel" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'bestPublicationsPlace'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ publicationTitle case="snake" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'best_publications_place'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-07-27 15:39:10 +00:00
|
|
|
it('should not create repeated characters when converting case to hyphen or snake', async function () {
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemNoRepeatedHyphens, { formatString: '{{ title case="hyphen" }}' }),
|
|
|
|
'no-repeated-hyphens'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemNoRepeatedHyphens, { formatString: '{{ publicationTitle case="hyphen" }}' }),
|
|
|
|
'no-repeated-hyphens'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemNoRepeatedUnderscores, { formatString: '{{ title case="snake" }}' }),
|
|
|
|
'no_repeated_underscores'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemNoRepeatedUnderscores, { formatString: '{{ publicationTitle case="snake" }}' }),
|
|
|
|
'no_repeated_underscores'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should accept itemType, attachmentTitle or any other field', function () {
|
2023-07-20 10:50:34 +00:00
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ itemType localize="true" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Journal Article'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ publicationTitle }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Best Publications Place'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ journalAbbreviation }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'BPP'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ publisher }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Awesome House'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, { formatString: '{{ volume }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'3'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ issue }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'42'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ pages }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'321'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemPatent, { formatString: '{{ number }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'HBK-8539b'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemPatent, { formatString: '{{ assignee }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Fast FooBar'
|
|
|
|
);
|
2024-07-27 15:39:10 +00:00
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ attachmentTitle }}', attachmentTitle: 'Full Text' }),
|
|
|
|
'Full Text'
|
|
|
|
);
|
2023-07-20 10:50:34 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("should support simple logic in template syntax", function () {
|
|
|
|
const template = '{{ if itemType == "journalArticle" }}j-{{ publicationTitle case="hyphen" }}{{ elseif itemType == "patent" }}p-{{ number case="hyphen" }}{{ else }}o-{{ title case="hyphen" }}{{ endif }}';
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, template), 'j-best-publications-place'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemPatent, template), 'p-hbk-8539b'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemManyAuthors, template), 'o-has-many-authors'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-07-27 15:39:10 +00:00
|
|
|
it("should skip missing fields", async function () {
|
2023-07-20 10:50:34 +00:00
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemIncomplete, { formatString: '{{ authors prefix = "a" suffix="-" }}{{ publicationTitle case="hyphen" suffix="-" }}{{ title }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'Incomplete'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should recognized base-mapped fields", function () {
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemBookSection, { formatString: '{{ bookTitle case="snake" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'book_title'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemBookSection, { formatString: '{{ publicationTitle case="snake" }}' }),
|
2023-07-20 10:50:34 +00:00
|
|
|
'book_title'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-07-27 15:39:10 +00:00
|
|
|
it("should trim spaces and remove new lines from the template string", function () {
|
2024-02-16 09:14:45 +00:00
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemBookSection, { formatString: ' {{ bookTitle case="snake" }}\n{{ bookTitle case="hyphen" prefix="-" }}' }),
|
|
|
|
'book_title-book-title'
|
2024-02-16 09:14:45 +00:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-07-16 05:59:07 +00:00
|
|
|
it("should suppress suffixes where they would create a repeat character", function () {
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ title suffix="-" }}{{ year prefix="-" }}' }),
|
2024-07-16 05:59:07 +00:00
|
|
|
'Lorem Ipsum-1975'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemSuffixes, { formatString: '{{ title prefix="-" suffix="-" }}{{ year }}' }),
|
2024-07-16 05:59:07 +00:00
|
|
|
'-Suffixes-1999'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemSuffixes, { formatString: '{{ title suffix="-" }}{{ year prefix="-" }}' }),
|
2024-07-16 05:59:07 +00:00
|
|
|
'-Suffixes-1999'
|
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemKeepHyphens, { formatString: '{{ title suffix="-" }}{{ year prefix="-" }}' }),
|
|
|
|
'keep--hyphens-1999'
|
2024-07-16 05:59:07 +00:00
|
|
|
);
|
2024-07-27 15:39:10 +00:00
|
|
|
// keep--hyphens is a title and should be kept unchanged but "keep" and "hyphens" are fields
|
2024-07-16 05:59:07 +00:00
|
|
|
// separated by prefixes and suffixes where repeated characters should be suppressed
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemKeepHyphens, { formatString: '{{ title suffix="-" }}{{ publicationTitle suffix="-" }}{{ issue prefix="-" }}' }),
|
|
|
|
'keep--hyphens-keep-hyphens'
|
2024-07-16 05:59:07 +00:00
|
|
|
);
|
2024-07-27 15:39:10 +00:00
|
|
|
// keep--hyphens is provided as literal part of the template and should be kept unchanged
|
|
|
|
// but "keep" and "hyphens" are fields separated by prefixes and suffixes where repeated
|
|
|
|
// characters should be suppressed. Finally "keep--hyphens" title is appended at the end
|
2024-07-16 05:59:07 +00:00
|
|
|
// which should also be kept as is.
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemKeepHyphens, { formatString: 'keep--hyphens-{{ publicationTitle prefix="-" suffix="-" }}{{ issue prefix="-" suffix="-" }}-keep--hyphens-{{ publicationTitle suffix="-" }}test{{ title prefix="-" }}' }),
|
|
|
|
'keep--hyphens-keep-hyphens-keep--hyphens-keep-test-keep--hyphens'
|
2024-07-16 05:59:07 +00:00
|
|
|
);
|
|
|
|
assert.equal(
|
2024-07-27 15:39:10 +00:00
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemSuffixes, { formatString: '{{ title prefix="/" suffix="\\" }}{{ year }}' }),
|
2024-07-16 05:59:07 +00:00
|
|
|
'-Suffixes-1999'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2024-07-27 15:39:10 +00:00
|
|
|
it("should be possible to test attachmentTitle", function () {
|
|
|
|
const template = `{{ if {{ attachmentTitle match="^(full.*|submitted.*|accepted.*)$" }} }}
|
|
|
|
{{ firstCreator suffix=" - " }}{{ year suffix=" - " }}{{ title truncate="100" }}
|
|
|
|
{{ else }}
|
|
|
|
{{ attachmentTitle replaceFrom="\\.pdf|\\.epub|\\.png" }}
|
|
|
|
{{ endif }}`;
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: template, attachmentTitle: 'Full Text' }),
|
|
|
|
'Barius and Pixelus - 1975 - Lorem Ipsum'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemBookSection, { formatString: template, attachmentTitle: 'Other Attachment.png' }),
|
|
|
|
'Other Attachment'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(itemBookSection, { formatString: `{{ attachmentTitle start = "6" truncate = "4" }}`, attachmentTitle: 'Other Attachment.png' }),
|
|
|
|
'Atta'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should perform regex in a case-insensitive way, unless configured otherwise", function () {
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ title match="lorem" }}' }),
|
|
|
|
'Lorem'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ title match="lorem" regexOpts="" }}' }),
|
|
|
|
'_' // template formatting results in an empty string, "_" is returned to make it a valid file name
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ title replaceFrom="lorem" replaceTo="Foobar" }}' }),
|
|
|
|
'Foobar Ipsum'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Attachments.getFileBaseNameFromItem(item, { formatString: '{{ title replaceFrom="lorem" replaceTo="foobar" regexOpts="" }}' }),
|
|
|
|
'Lorem Ipsum'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-08-04 09:58:15 +00:00
|
|
|
it("should convert old attachmentRenameFormatString to use new attachmentRenameTemplate syntax", function () {
|
2023-07-20 10:50:34 +00:00
|
|
|
assert.equal(
|
|
|
|
Zotero.Prefs.convertLegacyAttachmentRenameFormatString('{%c - }{%y - }{%t{50}}'),
|
|
|
|
'{{ firstCreator suffix=" - " }}{{ year suffix=" - " }}{{ title truncate="50" }}'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Prefs.convertLegacyAttachmentRenameFormatString('{ - %y - }'),
|
|
|
|
'{{ year prefix=" - " suffix=" - " }}'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Prefs.convertLegacyAttachmentRenameFormatString('{%y{2}00}'),
|
|
|
|
'{{ year truncate="2" suffix="00" }}'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Prefs.convertLegacyAttachmentRenameFormatString('{%c5 - }'),
|
|
|
|
'{{ firstCreator suffix="5 - " }}'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Prefs.convertLegacyAttachmentRenameFormatString('{%c-2 - }'),
|
|
|
|
'{{ firstCreator suffix="-2 - " }}'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Prefs.convertLegacyAttachmentRenameFormatString('{%t5 - }'),
|
|
|
|
'{{ title suffix="5 - " }}'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Prefs.convertLegacyAttachmentRenameFormatString('{++%t{10}--}'),
|
|
|
|
'{{ title truncate="10" prefix="++" suffix="--" }}'
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
Zotero.Prefs.convertLegacyAttachmentRenameFormatString('foo{%c}-{%t{10}}-{%y{2}00}'),
|
|
|
|
'foo{{ firstCreator }}-{{ title truncate="10" }}-{{ year truncate="2" suffix="00" }}'
|
|
|
|
);
|
|
|
|
});
|
2023-07-22 07:30:28 +00:00
|
|
|
|
|
|
|
it("should strip bidi isolates from firstCreator", async function () {
|
|
|
|
var item = createUnsavedDataObject('item',
|
|
|
|
{ creators: [{ name: 'Foo', creatorType: 'author' }, { name: 'Bar', creatorType: 'author' }] });
|
|
|
|
var str = Zotero.Attachments.getFileBaseNameFromItem(item);
|
|
|
|
assert.equal(str, Zotero.getString('general.andJoiner', ['Foo', 'Bar']) + ' - ');
|
|
|
|
});
|
2020-06-02 19:31:12 +00:00
|
|
|
});
|
|
|
|
|
2016-05-09 06:30:00 +00:00
|
|
|
describe("#getBaseDirectoryRelativePath()", function () {
|
2017-08-18 14:04:22 +00:00
|
|
|
it("should handle base directory at Windows drive root", function () {
|
|
|
|
Zotero.Prefs.set('baseAttachmentPath', "C:\\");
|
|
|
|
var path = Zotero.Attachments.getBaseDirectoryRelativePath("C:\\file.txt");
|
|
|
|
assert.equal(path, Zotero.Attachments.BASE_PATH_PLACEHOLDER + "file.txt");
|
|
|
|
});
|
|
|
|
|
2016-05-09 06:30:00 +00:00
|
|
|
it("should convert backslashes to forward slashes", function () {
|
|
|
|
Zotero.Prefs.set('baseAttachmentPath', "C:\\foo\\bar");
|
|
|
|
var path = Zotero.Attachments.getBaseDirectoryRelativePath("C:\\foo\\bar\\test\\file.txt");
|
|
|
|
assert.equal(path, Zotero.Attachments.BASE_PATH_PLACEHOLDER + "test/file.txt");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2015-08-09 08:52:14 +00:00
|
|
|
describe("#getTotalFileSize", function () {
|
|
|
|
it("should return the size for a single-file attachment", function* () {
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
|
|
|
|
// Create attachment and compare content
|
|
|
|
var item = yield Zotero.Attachments.importFromFile({
|
|
|
|
file: file
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.equal((yield Zotero.Attachments.getTotalFileSize(item)), file.fileSize);
|
|
|
|
})
|
|
|
|
})
|
2015-08-10 05:55:55 +00:00
|
|
|
|
|
|
|
describe("#hasMultipleFiles and #getNumFiles()", function () {
|
|
|
|
it("should return false and 1 for a single file", function* () {
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.png');
|
|
|
|
|
|
|
|
// Create attachment and compare content
|
|
|
|
var item = yield Zotero.Attachments.importFromFile({
|
|
|
|
file: file
|
|
|
|
});
|
|
|
|
|
|
|
|
assert.isFalse(yield Zotero.Attachments.hasMultipleFiles(item));
|
|
|
|
assert.equal((yield Zotero.Attachments.getNumFiles(item)), 1);
|
|
|
|
})
|
|
|
|
|
|
|
|
it("should return false and 1 for single HTML file with hidden file", function* () {
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.html');
|
|
|
|
|
|
|
|
// Create attachment and compare content
|
|
|
|
var item = yield Zotero.Attachments.importFromFile({
|
|
|
|
file: file
|
|
|
|
});
|
|
|
|
var path = OS.Path.join(OS.Path.dirname(item.getFilePath()), '.zotero-ft-cache');
|
|
|
|
yield Zotero.File.putContentsAsync(path, "");
|
|
|
|
|
|
|
|
assert.isFalse(yield Zotero.Attachments.hasMultipleFiles(item));
|
|
|
|
assert.equal((yield Zotero.Attachments.getNumFiles(item)), 1);
|
|
|
|
})
|
|
|
|
|
|
|
|
it("should return true and 2 for multiple files", function* () {
|
|
|
|
var file = getTestDataDirectory();
|
|
|
|
file.append('test.html');
|
|
|
|
|
|
|
|
// Create attachment and compare content
|
|
|
|
var item = yield Zotero.Attachments.importFromFile({
|
|
|
|
file: file
|
|
|
|
});
|
|
|
|
var path = OS.Path.join(OS.Path.dirname(item.getFilePath()), 'test.png');
|
|
|
|
yield Zotero.File.putContentsAsync(path, "");
|
|
|
|
|
|
|
|
assert.isTrue(yield Zotero.Attachments.hasMultipleFiles(item));
|
|
|
|
assert.equal((yield Zotero.Attachments.getNumFiles(item)), 2);
|
|
|
|
})
|
2016-12-12 08:26:35 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("#createDirectoryForItem()", function () {
|
|
|
|
it("should create missing directory", function* () {
|
|
|
|
var item = yield importFileAttachment('test.png');
|
|
|
|
var path = OS.Path.dirname(item.getFilePath());
|
|
|
|
yield OS.File.removeDir(path);
|
|
|
|
yield Zotero.Attachments.createDirectoryForItem(item);
|
|
|
|
assert.isTrue(yield OS.File.exists(path));
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should delete all existing files", function* () {
|
|
|
|
var item = yield importFileAttachment('test.html');
|
|
|
|
var path = OS.Path.dirname(item.getFilePath());
|
|
|
|
var files = ['a', 'b', 'c', 'd'];
|
|
|
|
for (let file of files) {
|
|
|
|
yield Zotero.File.putContentsAsync(OS.Path.join(path, file), file);
|
|
|
|
}
|
|
|
|
yield Zotero.Attachments.createDirectoryForItem(item);
|
|
|
|
assert.isTrue(yield Zotero.File.directoryIsEmpty(path));
|
|
|
|
assert.isTrue(yield OS.File.exists(path));
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should handle empty directory", function* () {
|
|
|
|
var item = yield importFileAttachment('test.png');
|
|
|
|
var file = item.getFilePath();
|
|
|
|
var dir = OS.Path.dirname(item.getFilePath());
|
|
|
|
yield OS.File.remove(file);
|
|
|
|
yield Zotero.Attachments.createDirectoryForItem(item);
|
|
|
|
assert.isTrue(yield OS.File.exists(dir));
|
|
|
|
});
|
|
|
|
});
|
2019-08-18 20:22:39 +00:00
|
|
|
|
|
|
|
describe("#convertLinkedFileToStoredFile()", function () {
|
|
|
|
it("should copy a linked file to a stored file", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var relatedItem = await createDataObject('item');
|
|
|
|
|
|
|
|
var originalFile = OS.Path.join(getTestDataDirectory().path, 'test.pdf');
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: originalFile,
|
|
|
|
title: 'Title',
|
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
attachment.setNote('Note');
|
|
|
|
attachment.setTags([{ tag: 'Tag' }]);
|
|
|
|
attachment.addRelatedItem(relatedItem);
|
|
|
|
await attachment.saveTx();
|
|
|
|
relatedItem.addRelatedItem(attachment);
|
|
|
|
await relatedItem.saveTx();
|
|
|
|
// Make sure we're indexed
|
|
|
|
await Zotero.Fulltext.indexItems([attachment.id]);
|
|
|
|
|
|
|
|
var newAttachment = await Zotero.Attachments.convertLinkedFileToStoredFile(attachment);
|
|
|
|
|
|
|
|
assert.isFalse(Zotero.Items.exists(attachment.id));
|
|
|
|
assert.isTrue(await OS.File.exists(originalFile));
|
|
|
|
assert.equal(newAttachment.attachmentLinkMode, Zotero.Attachments.LINK_MODE_IMPORTED_FILE);
|
|
|
|
assert.equal(newAttachment.attachmentContentType, 'application/pdf');
|
|
|
|
assert.isTrue(await newAttachment.fileExists());
|
|
|
|
assert.equal(newAttachment.getField('title'), 'Title');
|
2020-08-25 18:22:23 +00:00
|
|
|
assert.equal(newAttachment.note, 'Note');
|
2019-08-18 20:22:39 +00:00
|
|
|
assert.sameDeepMembers(newAttachment.getTags(), [{ tag: 'Tag' }]);
|
|
|
|
assert.sameMembers(newAttachment.relatedItems, [relatedItem.key]);
|
|
|
|
assert.sameMembers(relatedItem.relatedItems, [newAttachment.key]);
|
|
|
|
assert.isTrue(await OS.File.exists(Zotero.Fulltext.getItemCacheFile(newAttachment).path));
|
|
|
|
assert.equal(
|
|
|
|
await Zotero.Fulltext.getIndexedState(newAttachment),
|
|
|
|
Zotero.Fulltext.INDEX_STATE_INDEXED
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2021-03-12 11:26:59 +00:00
|
|
|
it("should move annotations to stored file", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
var relatedItem = await createDataObject('item');
|
|
|
|
|
|
|
|
var originalFile = OS.Path.join(getTestDataDirectory().path, 'test.pdf');
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: originalFile,
|
|
|
|
title: 'Title',
|
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
var annotation1 = await createAnnotation('highlight', attachment);
|
|
|
|
var annotation2 = await createAnnotation('note', attachment);
|
|
|
|
|
|
|
|
var newAttachment = await Zotero.Attachments.convertLinkedFileToStoredFile(attachment);
|
|
|
|
|
|
|
|
assert.isFalse(Zotero.Items.exists(attachment.id));
|
|
|
|
assert.isTrue(Zotero.Items.exists(annotation1.id));
|
|
|
|
assert.isTrue(Zotero.Items.exists(annotation2.id));
|
|
|
|
|
|
|
|
var annotations = newAttachment.getAnnotations();
|
|
|
|
assert.lengthOf(annotations, 2);
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2019-08-18 20:22:39 +00:00
|
|
|
it("should move a linked file to a stored file with `move: true`", async function () {
|
|
|
|
var item = await createDataObject('item');
|
|
|
|
|
|
|
|
var originalFile = OS.Path.join(Zotero.getTempDirectory().path, 'test.png');
|
|
|
|
await OS.File.copy(
|
|
|
|
OS.Path.join(getTestDataDirectory().path, 'test.png'),
|
|
|
|
originalFile
|
|
|
|
);
|
|
|
|
var attachment = await Zotero.Attachments.linkFromFile({
|
|
|
|
file: originalFile,
|
|
|
|
parentItemID: item.id
|
|
|
|
});
|
|
|
|
|
|
|
|
var newAttachment = await Zotero.Attachments.convertLinkedFileToStoredFile(
|
|
|
|
attachment,
|
|
|
|
{
|
|
|
|
move: true
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
assert.isFalse(Zotero.Items.exists(attachment.id));
|
|
|
|
assert.isFalse(await OS.File.exists(originalFile));
|
|
|
|
assert.equal(newAttachment.attachmentLinkMode, Zotero.Attachments.LINK_MODE_IMPORTED_FILE);
|
|
|
|
assert.isTrue(await newAttachment.fileExists());
|
|
|
|
});
|
|
|
|
});
|
2015-04-26 21:42:29 +00:00
|
|
|
})
|