Mailspring/internal_packages/thread-search/lib/search-query-subscription.es6
Evan Morikawa 313fdc2fd0 fix(search): Don't re-query search results on status change
If that method is fired immediately twice in a row, it's possible for
this._set to not re-instantiate causing the results to be blow away
2016-11-14 14:01:00 -08:00

195 lines
5.3 KiB
JavaScript

import _ from 'underscore'
import {
Actions,
NylasAPI,
Thread,
DatabaseStore,
ComponentRegistry,
FocusedContentStore,
MutableQuerySubscription,
} from 'nylas-exports'
import SearchActions from './search-actions'
const {LongConnectionStatus} = NylasAPI
class SearchQuerySubscription extends MutableQuerySubscription {
constructor(searchQuery, accountIds) {
super(null, {emitResultSet: true})
this._searchQuery = searchQuery
this._accountIds = accountIds
this.resetData()
this._connections = []
this._unsubscribers = [
FocusedContentStore.listen(::this.onFocusedContentChanged),
]
this._extDisposables = []
_.defer(() => this.performSearch())
}
replaceRange = () => {
// TODO
}
resetData() {
this._searchStartedAt = null
this._resultsReceivedAt = null
this._firstThreadSelectedAt = null
this._lastFocusedThread = null
this._focusedThreadCount = 0
}
performSearch() {
this._searchStartedAt = Date.now()
this.performLocalSearch()
this.performRemoteSearch()
this.performExtensionSearch()
}
performLocalSearch() {
let dbQuery = DatabaseStore.findAll(Thread).distinct()
if (this._accountIds.length === 1) {
dbQuery = dbQuery.where({accountId: this._accountIds[0]})
}
dbQuery = dbQuery
.search(this._searchQuery)
.order(Thread.attributes.lastMessageReceivedTimestamp.descending())
.limit(100)
dbQuery.then((results) => {
if (results.length > 0) {
this.replaceQuery(dbQuery)
}
})
}
_addThreadIdsToSearch(ids = []) {
const currentResults = this._set && this._set.ids().length > 0;
let searchIds = ids;
if (currentResults) {
const currentResultIds = this._set.ids()
searchIds = _.uniq(currentResultIds.concat(ids))
}
const dbQuery = (
DatabaseStore.findAll(Thread)
.where({id: searchIds})
.order(Thread.attributes.lastMessageReceivedTimestamp.descending())
)
this.replaceQuery(dbQuery)
}
performRemoteSearch() {
const accountsSearched = new Set()
let resultIds = []
const allAccountsSearched = () => accountsSearched.size === this._accountIds.length
const resultsReturned = () => {
// Don't emit a "result" until we have at least one thread to display.
// Otherwise it will show "No Results Found"
if (resultIds.length > 0 || allAccountsSearched()) {
this._addThreadIdsToSearch(resultIds)
}
}
this._connections = this._accountIds.map((accountId) => {
return NylasAPI.startLongConnection({
accountId,
path: `/threads/search/streaming?q=${encodeURIComponent(this._searchQuery)}`,
onResults: (results) => {
if (!this._resultsReceivedAt) {
this._resultsReceivedAt = Date.now()
}
const threads = results[0]
resultIds = resultIds.concat(_.pluck(threads, 'id'))
resultsReturned()
},
onStatusChanged: (status) => {
const hasClosed = [
LongConnectionStatus.Closed,
LongConnectionStatus.Ended,
].includes(status)
if (hasClosed) {
accountsSearched.add(accountId)
if (allAccountsSearched()) {
SearchActions.searchCompleted()
}
}
},
})
})
}
performExtensionSearch() {
const searchExtensions = ComponentRegistry.findComponentsMatching({
role: "SearchBarResults",
})
this._extDisposables = searchExtensions.map((ext) => {
return ext.observeThreadIdsForQuery(this._searchQuery)
.subscribe((ids = []) => {
const allIds = _.compact(_.flatten(ids))
if (allIds.length === 0) return;
this._addThreadIdsToSearch(allIds)
})
})
}
onFocusedContentChanged() {
const thread = FocusedContentStore.focused('thread')
const shouldRecordChange = (
thread &&
(this._lastFocusedThread || {}).id !== thread.id
)
if (shouldRecordChange) {
if (this._focusedThreadCount === 0) {
this._firstThreadSelectedAt = Date.now()
}
this._focusedThreadCount += 1
this._lastFocusedThread = thread
}
}
reportSearchMetrics() {
if (!this._searchStartedAt) {
return;
}
let timeToFirstServerResults = null;
let timeToFirstThreadSelected = null;
const timeInsideSearch = Math.round((Date.now() - this._searchStartedAt) / 1000)
const numItems = this._focusedThreadCount
const didSelectAnyThreads = numItems > 0
if (this._firstThreadSelectedAt) {
timeToFirstThreadSelected = Math.round((this._firstThreadSelectedAt - this._searchStartedAt) / 1000)
}
if (this._resultsReceivedAt) {
timeToFirstServerResults = Math.round((this._resultsReceivedAt - this._searchStartedAt) / 1000)
}
const data = {
numItems,
timeInsideSearch,
didSelectAnyThreads,
timeToFirstServerResults,
timeToFirstThreadSelected,
}
Actions.recordUserEvent("Search Performed", data)
this.resetData()
}
onLastCallbackRemoved() {
this.reportSearchMetrics();
this._connections.forEach((conn) => conn.end())
this._unsubscribers.forEach((unsub) => unsub())
this._extDisposables.forEach((disposable) => disposable.dispose())
}
}
export default SearchQuerySubscription