[Search] Overhaul generic search provider

Rewrite the generic search provider to use prototypes.  Increase performance
by utilizing the model service instead of the object service, and use a
simplified method of request queueing.
This commit is contained in:
Pete Richards 2015-10-16 15:26:46 -07:00
parent 099591ad2e
commit 78e5c0143b
3 changed files with 241 additions and 214 deletions

View File

@ -48,8 +48,7 @@
"depends": [ "depends": [
"$q", "$q",
"$log", "$log",
"throttle", "modelService",
"objectService",
"workerService", "workerService",
"topic", "topic",
"GENERIC_SEARCH_ROOTS" "GENERIC_SEARCH_ROOTS"

View File

@ -24,16 +24,15 @@
/** /**
* Module defining GenericSearchProvider. Created by shale on 07/16/2015. * Module defining GenericSearchProvider. Created by shale on 07/16/2015.
*/ */
define( define([
[],
function () { ], function (
) {
"use strict"; "use strict";
var DEFAULT_MAX_RESULTS = 100, var DEFAULT_MAX_RESULTS = 100,
DEFAULT_TIMEOUT = 1000, MAX_CONCURRENT_REQUESTS = 100;
MAX_CONCURRENT_REQUESTS = 100,
FLUSH_INTERVAL = 0,
stopTime;
/** /**
* A search service which searches through domain objects in * A search service which searches through domain objects in
@ -42,208 +41,241 @@ define(
* @constructor * @constructor
* @param $q Angular's $q, for promise consolidation. * @param $q Angular's $q, for promise consolidation.
* @param $log Anglar's $log, for logging. * @param $log Anglar's $log, for logging.
* @param {Function} throttle a function to throttle function invocations * @param {ModelService} modelService the model service.
* @param {ObjectService} objectService The service from which * @param {WorkerService} workerService the workerService.
* domain objects can be gotten. * @param {TopicService} topic the topic service.
* @param {WorkerService} workerService The service which allows * @param {Array} ROOTS An array of object Ids to begin indexing.
* more easy creation of web workers.
* @param {GENERIC_SEARCH_ROOTS} ROOTS An array of the root
* domain objects' IDs.
*/ */
function GenericSearchProvider($q, $log, throttle, objectService, workerService, topic, ROOTS) { function GenericSearchProvider($q, $log, modelService, workerService, topic, ROOTS) {
var indexed = {}, var provider = this;
pendingIndex = {},
pendingQueries = {},
toRequest = [],
worker = workerService.run('genericSearchWorker'),
mutationTopic = topic("mutation"),
indexingStarted = Date.now(),
pendingRequests = 0,
scheduleFlush;
this.worker = worker;
this.pendingQueries = pendingQueries;
this.$q = $q; this.$q = $q;
// pendingQueries is a dictionary with the key value pairs st this.$log = $log;
// the key is the timestamp and the value is the promise this.modelService = modelService;
function scheduleIdsForIndexing(ids) { this.indexedIds = {};
ids.forEach(function (id) { this.idsToIndex = [];
if (!indexed[id] && !pendingIndex[id]) { this.pendingIndex = {};
indexed[id] = true; this.pendingRequests = 0;
pendingIndex[id] = true;
toRequest.push(id);
}
});
scheduleFlush();
}
// Tell the web worker to add a domain object's model to its list of items. this.pendingQueries = {};
function indexItem(domainObject) {
var model = domainObject.getModel();
worker.postMessage({ this.worker = this.startWorker(workerService);
request: 'index',
model: model,
id: domainObject.getId()
});
if (Array.isArray(model.composition)) { ROOTS.forEach(function indexRoot(rootId) {
scheduleIdsForIndexing(model.composition); provider.scheduleForIndexing(rootId);
}
}
// Handles responses from the web worker. Namely, the results of
// a search request.
function handleResponse(event) {
if (event.data.request !== 'search') {
return; // no idea how to handle anything else.
}
var workerResults = event.data.results,
ids = Object.keys(workerResults);
objectService
.getObjects(ids)
.then(function (objects) {
var searchResults = Object
.keys(objects)
.map(function (id) {
return {
object: objects[id],
id: id,
score: workerResults[id].matchCount
};
});
// Resove the promise corresponding to this
pendingQueries[event.data.timestamp].resolve({
hits: searchResults,
total: searchResults.length,
timedOut: event.data.timedOut
});
});
}
function requestAndIndex(id) {
pendingRequests += 1;
objectService.getObjects([id]).then(function (objects) {
delete pendingIndex[id];
if (objects[id]) {
indexItem(objects[id]);
}
}, function () {
$log.warn("Failed to index domain object " + id);
}).then(function () {
pendingRequests -= 1;
scheduleFlush();
});
}
scheduleFlush = throttle(function flush() {
var batchSize =
Math.max(MAX_CONCURRENT_REQUESTS - pendingRequests, 0);
if (toRequest.length + pendingRequests < 1) {
$log.info([
'GenericSearch finished indexing after ',
((Date.now() - indexingStarted) / 1000).toFixed(2),
' seconds.'
].join(''));
} else {
toRequest.splice(-batchSize, batchSize)
.forEach(requestAndIndex);
}
}, FLUSH_INTERVAL);
worker.onmessage = handleResponse;
// Index the tree's contents once at the beginning
scheduleIdsForIndexing(ROOTS);
// Re-index items when they are mutated
mutationTopic.listen(function (domainObject) {
var id = domainObject.getId();
indexed[id] = false;
scheduleIdsForIndexing([id]);
}); });
} }
/** /**
* Searches through the filetree for domain objects which match * Query the search provider for results.
* the search term. This function is to be used as a fallback
* in the case where other search services are not avaliable.
* Returns a promise for a result object that has the format
* {hits: searchResult[], total: number, timedOut: boolean}
* where a searchResult has the format
* {id: string, object: domainObject, score: number}
* *
* Notes: * @param {String} input the string to search by.
* * The order of the results is not guarenteed. * @param {Number} timestamp part of the SearchProvider interface, ignored.
* * A domain object qualifies as a match for a search input if * @param {Number} maxResults max number of results to return.
* the object's name property contains any of the search terms * @returns {Promise} a promise for a modelResults object.
* (which are generated by splitting the input at spaces).
* * Scores are higher for matches that have more of the terms
* as substrings.
*
* @param input The text input that is the query.
* @param timestamp The time at which this function was called.
* This timestamp is used as a unique identifier for this
* query and the corresponding results.
* @param maxResults (optional) The maximum number of results
* that this function should return.
* @param timeout (optional) The time after which the search should
* stop calculations and return partial results.
*/ */
GenericSearchProvider.prototype.query = function query(input, timestamp, maxResults, timeout) { GenericSearchProvider.prototype.query = function (
var terms = [], input,
searchResults = [], timestamp,
pendingQueries = this.pendingQueries, maxResults
worker = this.worker, ) {
defer = this.$q.defer(); if (!maxResults) {
maxResults = DEFAULT_MAX_RESULTS;
}
// Tell the worker to search for items it has that match this searchInput. var queryId = this.dispatchSearch(input, maxResults),
// Takes the searchInput, as well as a max number of results (will return pendingQuery = this.$q.defer();
// less than that if there are fewer matches).
function workerSearch(searchInput, maxResults, timestamp, timeout) { this.pendingQueries[queryId] = pendingQuery;
var message = {
return pendingQuery.promise;
};
/**
* Creates a search worker and attaches handlers.
*
* @private
* @param workerService
* @returns worker the created search worker.
*/
GenericSearchProvider.prototype.startWorker = function (workerService) {
var worker = workerService.run('genericSearchWorker'),
provider = this;
worker.onmessage = function (messageEvent) {
provider.onWorkerMessage(messageEvent);
};
return worker;
};
/**
* Listen to the mutation topic and re-index objects when they are
* mutated.
*
* @private
* @param topic the topicService.
*/
GenericSearchProvider.prototype.indexOnMutation = function (topic) {
var mutationTopic = topic('mutation'),
provider = this;
mutationTopic.listen(function (mutatedObject) {
var id = mutatedObject.getId();
provider.indexed[id] = false;
provider.scheduleForIndexing(id);
});
};
/**
* Schedule an id to be indexed at a later date. If there are less
* pending requests then allowed, will kick off an indexing request.
*
* @private
* @param {String} id to be indexed.
*/
GenericSearchProvider.prototype.scheduleForIndexing = function (id) {
if (!this.indexedIds[id] && !this.pendingIndex[id]) {
this.indexedIds[id] = true;
this.pendingIndex[id] = true;
this.idsToIndex.push(id);
}
this.keepIndexing();
};
/**
* If there are less pending requests than concurrent requests, keep
* firing requests.
*
* @private
*/
GenericSearchProvider.prototype.keepIndexing = function () {
if (this.pendingRequests < MAX_CONCURRENT_REQUESTS) {
this.beginIndexRequest();
}
};
/**
* Pass an id and model to the worker to be indexed. If the model has
* composition, schedule those ids for later indexing.
*
* @private
* @param id a model id
* @param model a model
*/
GenericSearchProvider.prototype.index = function (id, model) {
var provider = this;
this.worker.postMessage({
request: 'index',
model: model,
id: id
});
if (Array.isArray(model.composition)) {
model.composition.forEach(function (id) {
provider.scheduleForIndexing(id);
});
}
};
/**
* Pulls an id from the indexing queue, loads it from the model service,
* and indexes it. Upon completion, tells the provider to keep
* indexing.
*
* @private
*/
GenericSearchProvider.prototype.beginIndexRequest = function () {
var idToIndex = this.idsToIndex.shift(),
provider = this;
if (!idToIndex) {
return;
}
this.pendingRequests += 1;
this.modelService
.getModels([idToIndex])
.then(function (models) {
delete provider.pendingIndex[idToIndex];
if (models[idToIndex]) {
provider.index(idToIndex, models[idToIndex]);
}
}, function () {
provider
.$log
.warn('Failed to index domain object ' + idToIndex);
})
.then(function () {
provider.keepIndexing();
});
};
/**
* Handle messages from the worker. Only really knows how to handle search
* results, which are parsed, transformed into a modelResult object, which
* is used to resolve the corresponding promise.
* @private
*/
GenericSearchProvider.prototype.onWorkerMessage = function (event) {
if (event.data.request !== 'search') {
return;
}
var pendingQuery = this.pendingQueries[event.data.queryId],
modelResults = {
timedOut: event.data.timedOut,
total: event.data.total
};
modelResults.hits = event.data.results.map(function (hit) {
return {
id: hit.item.id,
model: hit.item.model,
score: hit.matchCount
};
});
pendingQuery.resolve(modelResults);
delete this.pendingQueries[event.data.queryId];
};
/**
* @private
* @returns {Number} a unique, unusued query Id.
*/
GenericSearchProvider.prototype.makeQueryId = function () {
var queryId = Math.ceil(Math.random() * 100000);
while (this.pendingQueries[queryId]) {
queryId = Math.ceil(Math.random() * 100000);
}
return queryId;
};
/**
* Dispatch a search query to the worker and return a queryId.
*
* @private
* @returns {Number} a unique query Id for the query.
*/
GenericSearchProvider.prototype.dispatchSearch = function (
searchInput,
maxResults
) {
var queryId = this.makeQueryId();
this.worker.postMessage({
request: 'search', request: 'search',
input: searchInput, input: searchInput,
maxResults: maxResults, maxResults: maxResults,
timestamp: timestamp, queryId: queryId
timeout: timeout });
};
worker.postMessage(message);
}
// If the input is nonempty, do a search return queryId;
if (input !== '' && input !== undefined) {
// Allow us to access this promise later to resolve it later
pendingQueries[timestamp] = defer;
// Check to see if the user provided a maximum
// number of results to display
if (!maxResults) {
// Else, we provide a default value
maxResults = DEFAULT_MAX_RESULTS;
}
// Similarly, check if timeout was provided
if (!timeout) {
timeout = DEFAULT_TIMEOUT;
}
// Send the query to the worker
workerSearch(input, maxResults, timestamp, timeout);
return defer.promise;
} else {
// Otherwise return an empty result
return { hits: [], total: 0 };
}
}; };
return GenericSearchProvider; return GenericSearchProvider;
} });
);

View File

@ -79,8 +79,8 @@
request: 'search', request: 'search',
results: {}, results: {},
total: 0, total: 0,
timestamp: data.timestamp, timedOut: false,
timedOut: false queryId: data.queryId
}, },
matches = {}; matches = {};
@ -144,11 +144,7 @@
message.total = results.length; message.total = results.length;
message.results = results message.results = results
.slice(0, data.maxResults) .slice(0, data.maxResults);
.reduce(function arrayToObject(obj, match) {
obj[match.item.id] = match;
return obj;
}, {});
return message; return message;
} }