Improve search return values and performance
This commit is contained in:
parent
a3f6137dec
commit
326747c0ce
|
@ -1,5 +1,5 @@
|
||||||
const fetch = require('node-fetch');
|
const fetch = require('node-fetch');
|
||||||
const { Op, fn } = require('sequelize');
|
const { Op, fn, col } = require('sequelize');
|
||||||
|
|
||||||
const BooksController = require('../bookData');
|
const BooksController = require('../bookData');
|
||||||
const { quickSearchInventaire } = require('./Inventaire');
|
const { quickSearchInventaire } = require('./Inventaire');
|
||||||
|
@ -14,7 +14,7 @@ class SearchController {
|
||||||
constructor(sequelizeModels, searchTerm, options = defaultSearchOptions) {
|
constructor(sequelizeModels, searchTerm, options = defaultSearchOptions) {
|
||||||
this.models = sequelizeModels;
|
this.models = sequelizeModels;
|
||||||
this.searchTerm = searchTerm;
|
this.searchTerm = searchTerm;
|
||||||
this.searchBy = options.searchBy;
|
this.searchBy = options.searchBy.replace('title', 'name').replace('author', 'description');
|
||||||
this.source = options.source;
|
this.source = options.source;
|
||||||
this.lang = options.language;
|
this.lang = options.language;
|
||||||
}
|
}
|
||||||
|
@ -23,28 +23,40 @@ class SearchController {
|
||||||
return typeof this.searchTerm !== 'undefined' && this.searchTerm !== '';
|
return typeof this.searchTerm !== 'undefined' && this.searchTerm !== '';
|
||||||
}
|
}
|
||||||
|
|
||||||
get includeQuery() {
|
get bookReferenceSearchAttributes() {
|
||||||
return [
|
return {
|
||||||
{
|
include: [
|
||||||
model: this.models.Review,
|
{
|
||||||
where: { text: { [Op.not]: null } },
|
as: 'Interactions',
|
||||||
attributes: [[fn('COUNT', 'id'), 'total']], // Get the total number of text reviews
|
model: this.models.Review,
|
||||||
as: 'reviews',
|
attributes: ['id'],
|
||||||
},
|
required: false,
|
||||||
{
|
},
|
||||||
model: this.models.Review,
|
{
|
||||||
where: { rating: { [Op.not]: null } },
|
as: 'Reviews',
|
||||||
attributes: [[fn('AVG', 'rating'), 'average']], // Get the average star rating
|
model: this.models.Review,
|
||||||
as: 'rating',
|
attributes: ['id'],
|
||||||
},
|
required: false,
|
||||||
]
|
},
|
||||||
}
|
{
|
||||||
|
as: 'Ratings',
|
||||||
get orderQuery() {
|
model: this.models.Review,
|
||||||
return [{
|
attributes: ['rating'],
|
||||||
model: this.models.Review,
|
required: false,
|
||||||
attributes: [[fn('COUNT', 'id'), 'total']], // Get the total number of text reviews
|
},
|
||||||
}, 'total', 'DESC']; // Order references from most to least interaction
|
], // These are all subsets of Review model specified in BookReference associations
|
||||||
|
attributes: [
|
||||||
|
[col('BookReference.id'), 'id'],
|
||||||
|
'name',
|
||||||
|
'description',
|
||||||
|
'sources',
|
||||||
|
'covers',
|
||||||
|
[fn('COUNT', col('Interactions.id')), 'totalInteractions'],
|
||||||
|
[fn('COUNT', col('Reviews.id')), 'numReviews'],
|
||||||
|
[fn('AVG', col('Ratings.rating')), 'averageRating'],
|
||||||
|
],
|
||||||
|
order: [[col('totalInteractions'), 'DESC']],
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
async search() {
|
async search() {
|
||||||
|
@ -63,19 +75,18 @@ class SearchController {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add any search results that match refs with the same URI and delete from results array
|
// Add any search results that match refs with the same URI and delete from results array
|
||||||
searchResults.forEach((result, i) => {
|
const urisToCheck = searchResults.filter(
|
||||||
// If the result is not already in bookReferences
|
result => !bookReferences.some(ref => result.uri === ref.sources[this.source])
|
||||||
if (!bookReferences.some(ref => result.uri === ref.sources[this.source])) {
|
).map(result => result.uri);
|
||||||
// Check if the URI is already in references table
|
|
||||||
const reference = await this.searchReferencesBySourceCode(this.source, result.uri);
|
if (urisToCheck.length > 0) {
|
||||||
if (reference) {
|
const foundReferences = await this.searchReferencesBySourceCodes(this.source, urisToCheck);
|
||||||
bookReferences.push(reference);
|
return [
|
||||||
searchResults[i] = null;
|
...bookReferences,
|
||||||
}
|
...foundReferences,
|
||||||
} else { // If the result is already in references, null it out.
|
...searchResults.filter(result => !urisToCheck.includes(result.uri)),
|
||||||
searchResults[i] = null;
|
];
|
||||||
}
|
}
|
||||||
});
|
|
||||||
|
|
||||||
return [
|
return [
|
||||||
...bookReferences,
|
...bookReferences,
|
||||||
|
@ -84,24 +95,7 @@ class SearchController {
|
||||||
}
|
}
|
||||||
|
|
||||||
async searchReferences() {
|
async searchReferences() {
|
||||||
const { BookReference, Review } = this.models;
|
const { BookReference } = this.models;
|
||||||
|
|
||||||
// const includeQuery = [{
|
|
||||||
// model: Review,
|
|
||||||
// include: [
|
|
||||||
// {
|
|
||||||
// model: Reaction.scope('Review'),
|
|
||||||
// group: ['reactionType'],
|
|
||||||
// attributes: ['reactionType', [fn('COUNT', 'reactionType'), 'count']]
|
|
||||||
// },
|
|
||||||
// ],
|
|
||||||
// order: [{
|
|
||||||
// model: Reaction.scope('Review'),
|
|
||||||
// attributes: [[fn('COUNT', 'id'), 'total']],
|
|
||||||
// limit: 1,
|
|
||||||
// }, 'total', 'DESC'],
|
|
||||||
// limit: 5,
|
|
||||||
// }];
|
|
||||||
|
|
||||||
const exact = await BookReference.findAll({
|
const exact = await BookReference.findAll({
|
||||||
where: {
|
where: {
|
||||||
|
@ -114,14 +108,15 @@ class SearchController {
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
include: includeQuery(Review),
|
...this.bookReferenceSearchAttributes,
|
||||||
order: orderQuery(Review),
|
}).then( // Empty results give 1 empty model in an array, so filter those out
|
||||||
});
|
references => references.filter(ref => typeof ref.id !== 'undefined' && ref.id !== null)
|
||||||
|
);
|
||||||
|
|
||||||
if (exact.length > 0) {
|
if (exact.length > 0) {
|
||||||
return exact;
|
return exact;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If no exact matches are found, return any approximate ones.
|
// If no exact matches are found, return any approximate ones.
|
||||||
return await BookReference.findAll({
|
return await BookReference.findAll({
|
||||||
where: {
|
where: {
|
||||||
|
@ -136,9 +131,10 @@ class SearchController {
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
include: this.includeQuery,
|
...this.bookReferenceSearchAttributes,
|
||||||
order: this.orderQuery,
|
}).then( // Empty results give 1 empty model in an array, so filter those out
|
||||||
});
|
references => references.filter(ref => typeof ref.id !== 'undefined' && ref.id !== null)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async searchReferencesBySourceCode(source, sourceId) {
|
async searchReferencesBySourceCode(source, sourceId) {
|
||||||
|
@ -151,8 +147,26 @@ class SearchController {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
include: this.includeQuery,
|
...this.bookReferenceSearchAttributes,
|
||||||
});
|
}).then( // Empty results give 1 empty model in an array, so filter those out
|
||||||
|
references => references.filter(ref => typeof ref.id !== 'undefined' && ref.id !== null)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
async searchReferencesBySourceCodes(source, sourceIds) {
|
||||||
|
const sourceJSONKey = `"${source}"`; // Enable searching withing JSON column.
|
||||||
|
return await this.models.BookReference.findOne({
|
||||||
|
where: {
|
||||||
|
[Op.or]: sourceIds.map(sourceId => ({
|
||||||
|
source: {
|
||||||
|
[sourceJSONKey]: sourceId,
|
||||||
|
},
|
||||||
|
})),
|
||||||
|
},
|
||||||
|
...this.bookReferenceSearchAttributes,
|
||||||
|
}).then( // Empty results give 1 empty model in an array, so filter those out
|
||||||
|
references => references.filter(ref => typeof ref.id !== 'undefined' && ref.id !== null)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
Loading…
Reference in New Issue