2015-07-16 10:35:26 -07: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-08-04 13:03:34 -07:00
|
|
|
* Module defining ElasticSearchProvider. Created by shale on 07/16/2015.
|
2015-07-16 10:35:26 -07:00
|
|
|
*/
|
|
|
|
define(
|
|
|
|
[],
|
|
|
|
function () {
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
// JSLint doesn't like underscore-prefixed properties,
|
|
|
|
// so hide them here.
|
|
|
|
var ID = "_id",
|
|
|
|
SCORE = "_score",
|
2015-07-30 15:47:09 -07:00
|
|
|
DEFAULT_MAX_RESULTS = 100;
|
2015-07-16 10:35:26 -07:00
|
|
|
|
|
|
|
/**
|
2015-07-21 15:04:50 -07:00
|
|
|
* A search service which searches through domain objects in
|
|
|
|
* the filetree using ElasticSearch.
|
2015-07-16 10:35:26 -07:00
|
|
|
*
|
|
|
|
* @constructor
|
2015-07-21 15:14:07 -07:00
|
|
|
* @param $http Angular's $http service, for working with urls.
|
2015-07-21 15:04:50 -07:00
|
|
|
* @param {ObjectService} objectService the service from which
|
|
|
|
* domain objects can be gotten.
|
2015-08-24 13:14:43 -07:00
|
|
|
* @param ROOT the constant `ELASTIC_ROOT` which allows us to
|
2015-07-21 15:04:50 -07:00
|
|
|
* interact with ElasticSearch.
|
2015-07-16 10:35:26 -07:00
|
|
|
*/
|
2015-08-04 13:03:34 -07:00
|
|
|
function ElasticSearchProvider($http, objectService, ROOT) {
|
2015-08-17 11:20:23 -07:00
|
|
|
this.$http = $http;
|
|
|
|
this.objectService = objectService;
|
2015-08-24 13:14:43 -07:00
|
|
|
this.root = ROOT;
|
2015-08-17 11:20:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Searches through the filetree for domain objects using a search
|
|
|
|
* term. This is done through querying elasticsearch. 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 from highest to lowest score,
|
|
|
|
* as elsaticsearch determines them to be.
|
|
|
|
* * Uses the fuzziness operator to get more results.
|
|
|
|
* * More about this search's behavior at
|
|
|
|
* https://www.elastic.co/guide/en/elasticsearch/reference/current/search-uri-request.html
|
|
|
|
*
|
|
|
|
* @param searchTerm 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. Elasticsearch
|
|
|
|
* does not guarentee that this timeout will be strictly followed.
|
|
|
|
*/
|
2015-08-24 13:14:43 -07:00
|
|
|
ElasticSearchProvider.prototype.query = function query(searchTerm, timestamp, maxResults, timeout) {
|
2015-08-17 11:20:23 -07:00
|
|
|
var $http = this.$http,
|
|
|
|
objectService = this.objectService,
|
|
|
|
root = this.root,
|
|
|
|
esQuery;
|
2015-07-16 10:35:26 -07:00
|
|
|
|
|
|
|
function addFuzziness(searchTerm, editDistance) {
|
|
|
|
if (!editDistance) {
|
|
|
|
editDistance = '';
|
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-16 10:35:26 -07:00
|
|
|
return searchTerm.split(' ').map(function (s) {
|
2015-08-04 10:01:54 -07:00
|
|
|
// Don't add fuzziness for quoted strings
|
2015-08-03 09:38:30 -07:00
|
|
|
if (s.indexOf('"') !== -1) {
|
2015-07-16 10:35:26 -07:00
|
|
|
return s;
|
|
|
|
} else {
|
|
|
|
return s + '~' + editDistance;
|
|
|
|
}
|
|
|
|
}).join(' ');
|
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-16 10:35:26 -07:00
|
|
|
// Currently specific to elasticsearch
|
|
|
|
function processSearchTerm(searchTerm) {
|
2015-07-29 13:17:50 -07:00
|
|
|
var spaceIndex;
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-28 17:04:29 -07:00
|
|
|
// Cut out any extra spaces
|
2015-07-16 10:35:26 -07:00
|
|
|
while (searchTerm.substr(0, 1) === ' ') {
|
|
|
|
searchTerm = searchTerm.substring(1, searchTerm.length);
|
|
|
|
}
|
|
|
|
while (searchTerm.substr(searchTerm.length - 1, 1) === ' ') {
|
|
|
|
searchTerm = searchTerm.substring(0, searchTerm.length - 1);
|
|
|
|
}
|
2015-07-28 17:04:29 -07:00
|
|
|
spaceIndex = searchTerm.indexOf(' ');
|
|
|
|
while (spaceIndex !== -1) {
|
2015-07-29 13:17:50 -07:00
|
|
|
searchTerm = searchTerm.substring(0, spaceIndex) +
|
2015-08-17 11:20:23 -07:00
|
|
|
searchTerm.substring(spaceIndex + 1, searchTerm.length);
|
2015-07-28 17:04:29 -07:00
|
|
|
spaceIndex = searchTerm.indexOf(' ');
|
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-30 13:32:34 -07:00
|
|
|
// Add fuzziness for completeness
|
|
|
|
searchTerm = addFuzziness(searchTerm);
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-16 10:35:26 -07:00
|
|
|
return searchTerm;
|
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
|
|
|
|
// Processes results from the format that elasticsearch returns to
|
|
|
|
// a list of searchResult objects, then returns a result object
|
2015-08-04 10:01:54 -07:00
|
|
|
// (See documentation for query for object descriptions)
|
2015-07-21 15:14:07 -07:00
|
|
|
function processResults(rawResults, timestamp) {
|
2015-07-29 15:07:13 -07:00
|
|
|
var results = rawResults.data.hits.hits,
|
2015-07-16 10:35:26 -07:00
|
|
|
resultsLength = results.length,
|
|
|
|
ids = [],
|
2015-07-16 13:08:05 -07:00
|
|
|
scores = {},
|
2015-07-28 10:15:05 -07:00
|
|
|
searchResults = [],
|
|
|
|
i;
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-16 10:35:26 -07:00
|
|
|
// Get the result objects' IDs
|
2015-07-28 10:15:05 -07:00
|
|
|
for (i = 0; i < resultsLength; i += 1) {
|
2015-07-16 10:35:26 -07:00
|
|
|
ids.push(results[i][ID]);
|
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-16 13:08:05 -07:00
|
|
|
// Get the result objects' scores
|
2015-07-28 10:15:05 -07:00
|
|
|
for (i = 0; i < resultsLength; i += 1) {
|
|
|
|
scores[ids[i]] = results[i][SCORE];
|
2015-07-16 13:08:05 -07:00
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-16 10:35:26 -07:00
|
|
|
// Get the domain objects from their IDs
|
|
|
|
return objectService.getObjects(ids).then(function (objects) {
|
2015-07-28 10:15:05 -07:00
|
|
|
var j,
|
|
|
|
id;
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-28 10:15:05 -07:00
|
|
|
for (j = 0; j < resultsLength; j += 1) {
|
|
|
|
id = ids[j];
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-17 11:24:33 -07:00
|
|
|
// Include items we can get models for
|
2015-07-16 10:35:26 -07:00
|
|
|
if (objects[id].getModel) {
|
2015-07-21 11:17:45 -07:00
|
|
|
// Format the results as searchResult objects
|
|
|
|
searchResults.push({
|
|
|
|
id: id,
|
|
|
|
object: objects[id],
|
|
|
|
score: scores[id]
|
|
|
|
});
|
2015-07-16 10:35:26 -07:00
|
|
|
}
|
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
|
2015-07-30 13:54:56 -07:00
|
|
|
return {
|
|
|
|
hits: searchResults,
|
2015-07-30 16:35:26 -07:00
|
|
|
total: rawResults.data.hits.total,
|
|
|
|
timedOut: rawResults.data.timed_out
|
2015-07-30 13:54:56 -07:00
|
|
|
};
|
2015-07-16 10:35:26 -07:00
|
|
|
});
|
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the user input is empty, we want to have no search results.
|
|
|
|
if (searchTerm !== '' && searchTerm !== undefined) {
|
|
|
|
// Process the search term
|
|
|
|
searchTerm = processSearchTerm(searchTerm);
|
|
|
|
|
|
|
|
// Create the query to elasticsearch
|
|
|
|
esQuery = root + "/_search/?q=" + searchTerm +
|
|
|
|
"&size=" + maxResults;
|
|
|
|
if (timeout) {
|
|
|
|
esQuery += "&timeout=" + timeout;
|
2015-07-16 10:35:26 -07:00
|
|
|
}
|
2015-07-22 16:09:14 -07:00
|
|
|
|
2015-08-17 11:20:23 -07:00
|
|
|
// Get the data...
|
|
|
|
return this.$http({
|
|
|
|
method: "GET",
|
|
|
|
url: esQuery
|
|
|
|
}).then(function (rawResults) {
|
|
|
|
// ...then process the data
|
|
|
|
return processResults(rawResults, timestamp);
|
|
|
|
}, function (err) {
|
|
|
|
// In case of error, return nothing. (To prevent
|
|
|
|
// infinite loading time.)
|
2015-07-30 13:54:56 -07:00
|
|
|
return {hits: [], total: 0};
|
2015-08-17 11:20:23 -07:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
return {hits: [], total: 0};
|
2015-07-16 10:35:26 -07:00
|
|
|
}
|
2015-08-17 11:20:23 -07:00
|
|
|
};
|
2015-07-16 10:35:26 -07:00
|
|
|
|
|
|
|
|
2015-08-04 13:03:34 -07:00
|
|
|
return ElasticSearchProvider;
|
2015-07-16 10:35:26 -07:00
|
|
|
}
|
|
|
|
);
|