2015-07-16 17:35:26 +00:00
|
|
|
/*****************************************************************************
|
|
|
|
* Open MCT Web, Copyright (c) 2014-2015, United States Government
|
|
|
|
* as represented by the Administrator of the National Aeronautics and Space
|
|
|
|
* Administration. All rights reserved.
|
|
|
|
*
|
|
|
|
* Open MCT Web is licensed under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0.
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
|
|
|
*
|
|
|
|
* Open MCT Web includes source code licensed under additional open source
|
|
|
|
* licenses. See the Open Source Licenses file (LICENSES.md) included with
|
|
|
|
* this source code distribution or the Licensing information page available
|
|
|
|
* at runtime from the About dialog for additional information.
|
|
|
|
*****************************************************************************/
|
|
|
|
/*global define*/
|
|
|
|
|
|
|
|
/**
|
2015-07-17 18:24:33 +00:00
|
|
|
* Module defining GenericSearchProvider. Created by shale on 07/16/2015.
|
2015-07-16 17:35:26 +00:00
|
|
|
*/
|
|
|
|
define(
|
|
|
|
[],
|
|
|
|
function () {
|
|
|
|
"use strict";
|
2015-07-16 18:31:11 +00:00
|
|
|
|
2015-07-20 17:58:59 +00:00
|
|
|
var DEFAULT_MAX_RESULTS = 100,
|
|
|
|
DEFAULT_TIMEOUT = 1000,
|
|
|
|
stopTime;
|
2015-07-16 18:31:11 +00:00
|
|
|
|
2015-07-16 17:35:26 +00:00
|
|
|
/**
|
2015-07-20 17:58:59 +00:00
|
|
|
* A search service which searches through domain objects in
|
|
|
|
* the filetree without using external search implementations.
|
2015-07-16 17:35:26 +00:00
|
|
|
*
|
|
|
|
* @constructor
|
2015-08-04 17:01:54 +00:00
|
|
|
* @param $q Angular's $q, for promise consolidation.
|
2015-08-12 17:57:37 +00:00
|
|
|
* @param $timeout Angular's $timeout, for delayed function execution.
|
2015-08-04 17:01:54 +00:00
|
|
|
* @param {ObjectService} objectService The service from which
|
2015-07-20 17:58:59 +00:00
|
|
|
* domain objects can be gotten.
|
2015-08-04 17:01:54 +00:00
|
|
|
* @param {WorkerService} workerService The service which allows
|
2015-07-20 17:58:59 +00:00
|
|
|
* more easy creation of web workers.
|
2015-08-12 17:57:37 +00:00
|
|
|
* @param {GENERIC_SEARCH_ROOTS} ROOTS An array of the root
|
|
|
|
* domain objects' IDs.
|
2015-07-16 17:35:26 +00:00
|
|
|
*/
|
2015-08-12 16:42:21 +00:00
|
|
|
function GenericSearchProvider($q, $timeout, objectService, workerService, ROOTS) {
|
2015-08-17 18:20:23 +00:00
|
|
|
var indexed = {},
|
|
|
|
pendingQueries = {},
|
|
|
|
worker = workerService.run('genericSearchWorker');
|
|
|
|
|
|
|
|
this.worker = worker;
|
|
|
|
this.pendingQueries = pendingQueries;
|
|
|
|
this.$q = $q;
|
|
|
|
// pendingQueries is a dictionary with the key value pairs st
|
2015-07-29 22:07:13 +00:00
|
|
|
// the key is the timestamp and the value is the promise
|
|
|
|
|
2015-07-29 20:17:50 +00:00
|
|
|
// Tell the web worker to add a domain object's model to its list of items.
|
2015-07-21 16:58:19 +00:00
|
|
|
function indexItem(domainObject) {
|
2015-07-29 17:59:58 +00:00
|
|
|
var message;
|
|
|
|
|
|
|
|
// undefined check
|
|
|
|
if (domainObject && domainObject.getModel) {
|
|
|
|
// Using model instead of whole domain object because
|
|
|
|
// it's a JSON object.
|
|
|
|
message = {
|
|
|
|
request: 'index',
|
|
|
|
model: domainObject.getModel(),
|
|
|
|
id: domainObject.getId()
|
|
|
|
};
|
|
|
|
worker.postMessage(message);
|
|
|
|
}
|
2015-07-21 16:58:19 +00:00
|
|
|
}
|
|
|
|
|
2015-08-17 18:20:23 +00:00
|
|
|
|
2015-08-04 17:01:54 +00:00
|
|
|
// Handles responses from the web worker. Namely, the results of
|
|
|
|
// a search request.
|
2015-07-20 17:58:59 +00:00
|
|
|
function handleResponse(event) {
|
2015-07-29 20:17:50 +00:00
|
|
|
var ids = [],
|
2015-07-28 17:15:05 +00:00
|
|
|
id;
|
|
|
|
|
2015-07-29 20:17:50 +00:00
|
|
|
// If we have the results from a search
|
2015-07-21 18:14:10 +00:00
|
|
|
if (event.data.request === 'search') {
|
2015-07-21 19:34:08 +00:00
|
|
|
// Convert the ids given from the web worker into domain objects
|
2015-07-28 17:15:05 +00:00
|
|
|
for (id in event.data.results) {
|
2015-07-21 21:55:44 +00:00
|
|
|
ids.push(id);
|
2015-07-21 19:34:08 +00:00
|
|
|
}
|
|
|
|
objectService.getObjects(ids).then(function (objects) {
|
2015-07-30 20:54:56 +00:00
|
|
|
var searchResults = [],
|
2015-07-29 22:22:46 +00:00
|
|
|
id;
|
2015-07-28 18:33:29 +00:00
|
|
|
|
2015-08-04 17:01:54 +00:00
|
|
|
// Create searchResult objects
|
2015-07-28 17:15:05 +00:00
|
|
|
for (id in objects) {
|
2015-07-30 20:54:56 +00:00
|
|
|
searchResults.push({
|
2015-07-21 19:44:14 +00:00
|
|
|
object: objects[id],
|
2015-07-21 21:55:44 +00:00
|
|
|
id: id,
|
2015-07-30 20:19:19 +00:00
|
|
|
score: event.data.results[id]
|
2015-07-21 19:44:14 +00:00
|
|
|
});
|
2015-07-21 19:34:08 +00:00
|
|
|
}
|
2015-07-29 22:07:13 +00:00
|
|
|
|
|
|
|
// Resove the promise corresponding to this
|
2015-08-04 17:01:54 +00:00
|
|
|
pendingQueries[event.data.timestamp].resolve({
|
|
|
|
hits: searchResults,
|
|
|
|
total: event.data.total,
|
|
|
|
timedOut: event.data.timedOut
|
|
|
|
});
|
2015-07-21 19:34:08 +00:00
|
|
|
});
|
2015-07-21 18:14:10 +00:00
|
|
|
}
|
2015-07-20 17:58:59 +00:00
|
|
|
}
|
2015-07-16 17:35:26 +00:00
|
|
|
|
2015-08-06 16:51:09 +00:00
|
|
|
// Helper function for getItems(). Indexes the tree.
|
|
|
|
function indexItems(nodes) {
|
2015-08-06 16:55:08 +00:00
|
|
|
nodes.forEach(function (node) {
|
2015-08-06 21:41:47 +00:00
|
|
|
var id = node && node.getId && node.getId();
|
2015-08-06 16:51:09 +00:00
|
|
|
|
2015-08-12 17:57:37 +00:00
|
|
|
// If we have already indexed this item, stop here
|
|
|
|
if (indexed[id]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Index each item with the web worker
|
|
|
|
indexItem(node);
|
|
|
|
indexed[id] = true;
|
|
|
|
|
|
|
|
|
|
|
|
// If this node has children, index those
|
|
|
|
if (node && node.hasCapability && node.hasCapability('composition')) {
|
|
|
|
// Make sure that this is async, so doesn't block up page
|
|
|
|
$timeout(function () {
|
|
|
|
// Get the children...
|
|
|
|
node.useCapability('composition').then(function (children) {
|
|
|
|
$timeout(function () {
|
|
|
|
// ... then index the children
|
|
|
|
if (children.constructor === Array) {
|
|
|
|
indexItems(children);
|
|
|
|
} else {
|
|
|
|
indexItems([children]);
|
|
|
|
}
|
|
|
|
}, 0);
|
|
|
|
});
|
|
|
|
}, 0);
|
2015-08-06 16:51:09 +00:00
|
|
|
}
|
2015-08-06 21:41:47 +00:00
|
|
|
|
|
|
|
// Watch for changes to this item, in case it gets new children
|
|
|
|
if (node && node.hasCapability && node.hasCapability('mutation')) {
|
|
|
|
node.getCapability('mutation').listen(function (listener) {
|
|
|
|
if (listener && listener.composition) {
|
|
|
|
// If the node was mutated to have children, get the child domain objects
|
|
|
|
objectService.getObjects(listener.composition).then(function (objectsById) {
|
|
|
|
var objects = [],
|
|
|
|
id;
|
|
|
|
|
|
|
|
// Get each of the domain objects in objectsById
|
|
|
|
for (id in objectsById) {
|
|
|
|
objects.push(objectsById[id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
indexItems(objects);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2015-08-06 16:55:08 +00:00
|
|
|
});
|
2015-07-16 17:35:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Converts the filetree into a list
|
2015-08-04 20:16:29 +00:00
|
|
|
function getItems() {
|
2015-07-22 23:21:34 +00:00
|
|
|
// Aquire root objects
|
2015-08-04 20:16:29 +00:00
|
|
|
objectService.getObjects(ROOTS).then(function (objectsById) {
|
2015-07-28 17:15:05 +00:00
|
|
|
var objects = [],
|
|
|
|
id;
|
2015-07-20 17:58:59 +00:00
|
|
|
|
2015-08-04 17:01:54 +00:00
|
|
|
// Get each of the domain objects in objectsById
|
2015-07-28 17:15:05 +00:00
|
|
|
for (id in objectsById) {
|
2015-07-22 23:21:34 +00:00
|
|
|
objects.push(objectsById[id]);
|
|
|
|
}
|
|
|
|
|
2015-07-29 17:59:58 +00:00
|
|
|
// Index all of the roots' descendents
|
2015-08-06 16:51:09 +00:00
|
|
|
indexItems(objects);
|
2015-07-16 17:35:26 +00:00
|
|
|
});
|
|
|
|
}
|
2015-08-11 20:52:23 +00:00
|
|
|
|
2015-08-17 18:20:23 +00:00
|
|
|
worker.onmessage = handleResponse;
|
2015-08-11 20:52:23 +00:00
|
|
|
|
2015-07-29 17:26:24 +00:00
|
|
|
// Index the tree's contents once at the beginning
|
2015-07-29 20:17:50 +00:00
|
|
|
getItems();
|
2015-07-16 17:35:26 +00:00
|
|
|
}
|
|
|
|
|
2015-08-17 18:20:23 +00:00
|
|
|
/**
|
|
|
|
* Searches through the filetree for domain objects which match
|
|
|
|
* 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:
|
|
|
|
* * The order of the results is not guarenteed.
|
|
|
|
* * A domain object qualifies as a match for a search input if
|
|
|
|
* the object's name property contains any of the search terms
|
|
|
|
* (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) {
|
|
|
|
var terms = [],
|
|
|
|
searchResults = [],
|
|
|
|
pendingQueries = this.pendingQueries,
|
|
|
|
worker = this.worker,
|
|
|
|
defer = this.$q.defer();
|
|
|
|
|
|
|
|
// Tell the worker to search for items it has that match this searchInput.
|
|
|
|
// Takes the searchInput, as well as a max number of results (will return
|
|
|
|
// less than that if there are fewer matches).
|
|
|
|
function workerSearch(searchInput, maxResults, timestamp, timeout) {
|
|
|
|
var message = {
|
|
|
|
request: 'search',
|
|
|
|
input: searchInput,
|
|
|
|
maxNumber: maxResults,
|
|
|
|
timestamp: timestamp,
|
|
|
|
timeout: timeout
|
|
|
|
};
|
|
|
|
worker.postMessage(message);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the input is nonempty, do a search
|
|
|
|
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 };
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-07-16 17:35:26 +00:00
|
|
|
|
2015-07-17 18:24:33 +00:00
|
|
|
return GenericSearchProvider;
|
2015-07-16 17:35:26 +00:00
|
|
|
}
|
|
|
|
);
|