forked from zotero/translators
-
Notifications
You must be signed in to change notification settings - Fork 0
/
CSL JSON.js
172 lines (158 loc) · 5.22 KB
/
CSL JSON.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
{
"translatorID": "bc03b4fe-436d-4a1f-ba59-de4d2d7a63f7",
"label": "CSL JSON",
"creator": "Simon Kornblith",
"target": "json",
"minVersion": "4.0.27",
"maxVersion": "",
"priority": 100,
"configOptions": {
"async": true
},
"inRepository": true,
"translatorType": 3,
"lastUpdated": "2020-09-23 04:40:23"
}
function parseInput() {
var str, json = "";
// Read in the whole file at once, since we can't easily parse a JSON stream. The
// chunk size here is pretty arbitrary, although larger chunk sizes may be marginally
// faster. We set it to 1MB.
while ((str = Z.read(1048576)) !== false) json += str;
try {
return JSON.parse(json);
} catch(e) {
Zotero.debug(e);
}
}
function detectImport() {
const CSL_TYPES = {"article":true, "article-journal":true, "article-magazine":true,
"article-newspaper":true, "bill":true, "book":true, "broadcast":true,
"chapter":true, "dataset":true, "entry":true, "entry-dictionary":true,
"entry-encyclopedia":true, "figure":true, "graphic":true, "interview":true,
"legal_case":true, "legislation":true, "manuscript":true, "map":true,
"motion_picture":true, "musical_score":true, "pamphlet":true,
"paper-conference":true, "patent":true, "personal_communication":true,
"post":true, "post-weblog":true, "report":true, "review":true, "review-book":true,
"song":true, "speech":true, "thesis":true, "treaty":true, "webpage":true};
var parsedData = parseInput();
if (!parsedData) return false;
if (typeof parsedData !== "object") return false;
if (!(parsedData instanceof Array)) parsedData = [parsedData];
for (var i=0; i<parsedData.length; i++) {
var item = parsedData[i];
if (typeof item !== "object" || !item.type || !(item.type in CSL_TYPES)) {
return false;
}
}
return true;
}
function doImport() {
if (typeof Promise == 'undefined') {
startImport(
function () {},
function (e) {
throw e;
}
);
}
else {
return new Promise(function (resolve, reject) {
startImport(resolve, reject);
});
}
}
function startImport(resolve, reject) {
try {
var parsedData = parseInput();
if (!parsedData) resolve();
if (!Array.isArray(parsedData)) parsedData = [parsedData];
importNext(parsedData, resolve, reject);
}
catch (e) {
reject (e);
}
}
function importNext(data, resolve, reject) {
try {
var d;
while (d = data.shift()) {
var item = new Z.Item();
// Default to 'article' (Document) if no type given. 'type' is required in CSL-JSON,
// but some DOI registration agencies provide bad data, and this is better than failing.
// (itemFromCSLJSON() will already default to 'article' for unknown 'type' values.)
//
// Technically this should go in the DOI Content Negotation translator, but it's easier
// to do this here after the JSON has been parsed, and it might benefit other translators.
//
// This is just for imports from other translators. File/clipboard imports without
// 'type' still won't work, because a valid 'type' is required in detectImport().
//
// https://forums.zotero.org/discussion/85273/error-importing-dois-via-add-item-by-identifier
if (!d.type) {
d.type = 'article';
}
ZU.itemFromCSLJSON(item, d);
var maybePromise = item.complete();
if (maybePromise) {
maybePromise.then(function () {
importNext(data, resolve, reject);
});
return;
}
}
}
catch (e) {
reject(e);
}
resolve();
}
function doExport() {
var item, data = [];
while (item = Z.nextItem()) {
if (item.extra) {
item.extra = item.extra.replace(/(?:^|\n)citation key\s*:\s*([^\s]+)(?:\n|$)/i, (m, citationKey) => {
item.citationKey = citationKey;
return '\n';
}).trim();
}
var cslItem = ZU.itemToCSLJSON(item);
if (item.citationKey) cslItem.id = item.citationKey;
data.push(cslItem);
}
Z.write(JSON.stringify(data, null, "\t"));
}
/** BEGIN TEST CASES **/
var testCases = [
{
"type": "import",
"input": "[\n\t{\n\t\t\"id\": \"http://zotero.org/users/96641/items/BDQRTS3T\",\n\t\t\"type\": \"book\",\n\t\t\"title\": \"Stochastic biomathematical models: With applications to neuronal modeling\",\n\t\t\"collection-title\": \"Lecture notes in mathematics\",\n\t\t\"publisher\": \"Springer\",\n\t\t\"publisher-place\": \"Heidelberg\",\n\t\t\"volume\": \"2058\",\n\t\t\"number-of-pages\": \"206\",\n\t\t\"event-place\": \"Heidelberg\",\n\t\t\"ISBN\": \"978-3-642-32156-6\",\n\t\t\"language\": \"en\",\n\t\t\"author\": [\n\t\t\t{\n\t\t\t\t\"family\": \"Bachar\",\n\t\t\t\t\"given\": \"Mostafa\"\n\t\t\t}\n\t\t],\n\t\t\"issued\": {\n\t\t\t\"date-parts\": [\n\t\t\t\t[\n\t\t\t\t\t\"2013\",\n\t\t\t\t\t1,\n\t\t\t\t\t1\n\t\t\t\t]\n\t\t\t]\n\t\t}\n\t}\n]",
"items": [
{
"itemType": "book",
"title": "Stochastic biomathematical models: With applications to neuronal modeling",
"creators": [
{
"lastName": "Bachar",
"firstName": "Mostafa",
"creatorType": "author"
}
],
"date": "January 1, 2013",
"ISBN": "978-3-642-32156-6",
"itemID": "http://zotero.org/users/96641/items/BDQRTS3T",
"language": "en",
"numPages": "206",
"place": "Heidelberg",
"publisher": "Springer",
"series": "Lecture notes in mathematics",
"volume": "2058",
"attachments": [],
"tags": [],
"notes": [],
"seeAlso": []
}
]
}
]
/** END TEST CASES **/