Fix error trying to generate report for many items

When adding many search conditions (e.g., when matching many items with the
`key` condition), the query can fail due to either the bound parameter limit or
the expression tree size limit.

To avoid this, add support for an 'inlineFilter' property on search conditions
when using the 'is' or 'isNot' operator. 'inlineFilter' is a function that
returns a quoted value suitable for direct embedding in the SQL statement, or
false if not valid. Multiple consecutive conditions for the same 'inlineFilter'
field are combined into an `IN (x, y, z)` condition.
This commit is contained in:
Dan Stillman 2017-01-21 03:38:36 -05:00
parent dcd1da70af
commit 9b247ebba7
4 changed files with 99 additions and 39 deletions

View file

@ -96,17 +96,14 @@ Zotero.API = {
s.addCondition('key', 'is', params.objectKey); s.addCondition('key', 'is', params.objectKey);
} }
else if (params.objectID) { else if (params.objectID) {
Zotero.debug('adding ' + params.objectID);
s.addCondition('itemID', 'is', params.objectID); s.addCondition('itemID', 'is', params.objectID);
} }
if (params.itemKey) { if (params.itemKey) {
s.addCondition('blockStart');
for (let i=0; i<params.itemKey.length; i++) { for (let i=0; i<params.itemKey.length; i++) {
let itemKey = params.itemKey[i]; let itemKey = params.itemKey[i];
s.addCondition('key', 'is', itemKey); s.addCondition('key', 'is', itemKey);
} }
s.addCondition('blockEnd');
} }
// Display all top-level items // Display all top-level items

View file

@ -925,61 +925,78 @@ Zotero.Search.prototype._buildQuery = Zotero.Promise.coroutine(function* () {
var conditions = []; var conditions = [];
for (var i in this._conditions){ let lastCondition;
var data = Zotero.SearchConditions.get(this._conditions[i]['condition']); for (let condition of Object.values(this._conditions)) {
let name = condition.condition;
let conditionData = Zotero.SearchConditions.get(name);
// Has a table (or 'savedSearch', which doesn't have a table but isn't special) // Has a table (or 'savedSearch', which doesn't have a table but isn't special)
if (data.table || data.name == 'savedSearch' || data.name == 'tempTable') { if (conditionData.table || name == 'savedSearch' || name == 'tempTable') {
conditions.push({ // For conditions with an inline filter using 'is'/'isNot', combine with last condition
name: data['name'], // if the same
alias: data['name']!=this._conditions[i]['condition'] if (lastCondition
? this._conditions[i]['condition'] : false, && name == lastCondition.name
table: data['table'], && condition.operator.startsWith('is')
field: data['field'], && condition.operator == lastCondition.operator
operator: this._conditions[i]['operator'], && conditionData.inlineFilter) {
value: this._conditions[i]['value'], if (!Array.isArray(lastCondition.value)) {
flags: data['flags'], lastCondition.value = [lastCondition.value];
required: this._conditions[i]['required'] }
}); lastCondition.value.push(condition.value);
continue;
}
lastCondition = {
name,
alias: conditionData.name != name ? name : false,
table: conditionData.table,
field: conditionData.field,
operator: condition.operator,
value: condition.value,
flags: conditionData.flags,
required: condition.required,
inlineFilter: conditionData.inlineFilter
};
conditions.push(lastCondition);
this._hasPrimaryConditions = true; this._hasPrimaryConditions = true;
} }
// Handle special conditions // Handle special conditions
else { else {
switch (data['name']){ switch (conditionData.name) {
case 'deleted': case 'deleted':
var deleted = this._conditions[i].operator == 'true'; var deleted = condition.operator == 'true';
continue; continue;
case 'noChildren': case 'noChildren':
var noChildren = this._conditions[i]['operator']=='true'; var noChildren = condition.operator == 'true';
continue; continue;
case 'includeParentsAndChildren': case 'includeParentsAndChildren':
var includeParentsAndChildren = this._conditions[i]['operator'] == 'true'; var includeParentsAndChildren = condition.operator == 'true';
continue; continue;
case 'includeParents': case 'includeParents':
var includeParents = this._conditions[i]['operator'] == 'true'; var includeParents = condition.operator == 'true';
continue; continue;
case 'includeChildren': case 'includeChildren':
var includeChildren = this._conditions[i]['operator'] == 'true'; var includeChildren = condition.operator == 'true';
continue; continue;
case 'unfiled': case 'unfiled':
var unfiled = this._conditions[i]['operator'] == 'true'; var unfiled = condition.operator == 'true';
continue; continue;
// Search subcollections // Search subcollections
case 'recursive': case 'recursive':
var recursive = this._conditions[i]['operator']=='true'; var recursive = condition.operator == 'true';
continue; continue;
// Join mode ('any' or 'all') // Join mode ('any' or 'all')
case 'joinMode': case 'joinMode':
var joinMode = this._conditions[i]['operator'].toUpperCase(); var joinMode = condition.operator.toUpperCase();
continue; continue;
case 'fulltextContent': case 'fulltextContent':
@ -995,7 +1012,7 @@ Zotero.Search.prototype._buildQuery = Zotero.Promise.coroutine(function* () {
continue; continue;
} }
throw ('Unhandled special condition ' + this._conditions[i]['condition']); throw new Error('Unhandled special condition ' + name);
} }
} }
@ -1462,20 +1479,45 @@ Zotero.Search.prototype._buildQuery = Zotero.Promise.coroutine(function* () {
case 'is': case 'is':
case 'isNot': // excluded with NOT IN above case 'isNot': // excluded with NOT IN above
// Automatically cast values which might // If inline filter is available, embed value directly to get around
// have been stored as integers // the max bound parameter limit
if (condition.value && typeof condition.value == 'string' if (condition.inlineFilter) {
&& condition.value.match(/^[1-9]+[0-9]*$/)) { let src = Array.isArray(condition.value)
condSQL += ' LIKE ?'; ? condition.value : [condition.value];
} let values = [];
else if (condition.value === null) {
condSQL += ' IS NULL'; for (let val of src) {
break; val = condition.inlineFilter(val);
if (val) {
values.push(val);
}
else {
Zotero.logError(`${val} is not a valid `
+ `'${condition.field}' value -- skipping`);
continue;
}
}
condSQL += values.length > 1
? ` IN (${values.join(', ')})`
: `=${values[0]}`;
} }
else { else {
condSQL += '=?'; // Automatically cast values which might
// have been stored as integers
if (condition.value && typeof condition.value == 'string'
&& condition.value.match(/^[1-9]+[0-9]*$/)) {
condSQL += ' LIKE ?';
}
else if (condition.value === null) {
condSQL += ' IS NULL';
break;
}
else {
condSQL += '=?';
}
condSQLParams.push(condition['value']);
} }
condSQLParams.push(condition['value']);
break; break;
case 'beginsWith': case 'beginsWith':

View file

@ -450,7 +450,17 @@ Zotero.SearchConditions = new function(){
table: 'items', table: 'items',
field: 'key', field: 'key',
special: true, special: true,
noLoad: true noLoad: true,
inlineFilter: function (val) {
try {
val = Zotero.DataObjectUtilities.checkKey(val);
if (val) return `'${val}'`;
}
catch (e) {
Zotero.logError(e);
}
return false;
}
}, },
{ {

View file

@ -224,6 +224,17 @@ describe("Zotero.Search", function() {
}); });
}); });
describe("key", function () {
it("should allow more than max bound parameters", function* () {
let s = new Zotero.Search();
let max = Zotero.DB.MAX_BOUND_PARAMETERS + 100;
for (let i = 0; i < max; i++) {
s.addCondition('key', 'is', Zotero.DataObjectUtilities.generateKey());
}
yield s.search();
});
});
describe("savedSearch", function () { describe("savedSearch", function () {
it("should return items in the saved search", function* () { it("should return items in the saved search", function* () {
var search = yield createDataObject('search'); var search = yield createDataObject('search');