mirror of https://github.com/perkeep/perkeep.git
280 lines
8.7 KiB
JavaScript
280 lines
8.7 KiB
JavaScript
/*
|
|
Copyright 2013 Google Inc.
|
|
|
|
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.
|
|
*/
|
|
|
|
goog.provide('cam.SearchSession');
|
|
|
|
goog.require('goog.events.EventTarget');
|
|
goog.require('goog.Uri');
|
|
goog.require('goog.Uri.QueryData');
|
|
goog.require('goog.uri.utils');
|
|
|
|
goog.require('cam.ServerConnection');
|
|
|
|
// A search session is a standing query that notifies you when results change. It caches previous results and handles merging new data as it is received. It does not tell you _what_ changed; clients must reconcile as they see fit.
|
|
//
|
|
// TODO(aa): Only deltas should be sent from server to client
|
|
// TODO(aa): Need some way to avoid the duplicate query when websocket starts. Ideas:
|
|
// - Initial XHR query can also specify tag. This tag times out if not used rapidly. Send this same tag in socket query.
|
|
// - Socket assumes that client already has first batch of results (slightly racey though)
|
|
// - Prefer to use socket on client-side, test whether it works and fall back to XHR if not.
|
|
cam.SearchSession = function(connection, currentUri, query, opt_aroundBlobref) {
|
|
goog.base(this);
|
|
|
|
this.connection_ = connection;
|
|
this.currentUri_ = currentUri;
|
|
this.initSocketUri_(currentUri);
|
|
this.hasSocketError_ = false;
|
|
this.query_ = query;
|
|
this.around_ = opt_aroundBlobref;
|
|
this.tag_ = 'q' + (this.constructor.instanceCount_++);
|
|
this.continuation_ = this.getContinuation_(this.constructor.SEARCH_SESSION_CHANGE_TYPE.NEW);
|
|
this.socket_ = null;
|
|
this.supportsWebSocket_ = false;
|
|
this.isComplete_ = false;
|
|
|
|
this.resetData_();
|
|
};
|
|
goog.inherits(cam.SearchSession, goog.events.EventTarget);
|
|
|
|
// We fire this event when the data changes in any way.
|
|
cam.SearchSession.SEARCH_SESSION_CHANGED = 'search-session-change';
|
|
|
|
// We fire this event when the search session receives general server status data.
|
|
cam.SearchSession.SEARCH_SESSION_STATUS = 'search-session-status';
|
|
|
|
// We fire this event when the search session encounters an error.
|
|
cam.SearchSession.SEARCH_SESSION_ERROR = 'search-session-error';
|
|
|
|
// TODO(aa): This is only used by BlobItemContainer. Once we switch over to BlobItemContainerReact completely, it can be removed.
|
|
cam.SearchSession.SEARCH_SESSION_CHANGE_TYPE = {
|
|
NEW: 1,
|
|
APPEND: 2,
|
|
UPDATE: 3
|
|
};
|
|
|
|
cam.SearchSession.PAGE_SIZE_ = 50;
|
|
|
|
cam.SearchSession.instanceCount_ = 0;
|
|
|
|
cam.SearchSession.prototype.getQuery = function() {
|
|
return this.query_;
|
|
};
|
|
|
|
cam.SearchSession.prototype.getAround = function() {
|
|
return this.around_;
|
|
};
|
|
|
|
// Returns all the data we currently have loaded.
|
|
// It is guaranteed to return the following properties:
|
|
// blobs // non-null
|
|
// description
|
|
// description.meta
|
|
cam.SearchSession.prototype.getCurrentResults = function() {
|
|
return this.data_;
|
|
};
|
|
|
|
cam.SearchSession.prototype.hasSocketError = function() {
|
|
return this.hasSocketError_;
|
|
};
|
|
|
|
// Loads the next page of data. This is safe to call while a load is in progress; multiple calls for the same page will be collapsed. The SEARCH_SESSION_CHANGED event will be dispatched when the new data is available.
|
|
cam.SearchSession.prototype.loadMoreResults = function() {
|
|
if (!this.continuation_) {
|
|
return;
|
|
}
|
|
|
|
var c = this.continuation_;
|
|
this.continuation_ = null;
|
|
c();
|
|
};
|
|
|
|
// Returns true if it is known that all data which can be loaded for this query has been.
|
|
cam.SearchSession.prototype.isComplete = function() {
|
|
return this.isComplete_;
|
|
}
|
|
|
|
cam.SearchSession.prototype.supportsChangeNotifications = function() {
|
|
return this.supportsWebSocket_;
|
|
};
|
|
|
|
cam.SearchSession.prototype.refreshIfNecessary = function() {
|
|
if (this.supportsWebSocket_) {
|
|
return;
|
|
}
|
|
|
|
this.continuation_ = this.getContinuation_(this.constructor.SEARCH_SESSION_CHANGE_TYPE.UPDATE, null, Math.max(this.data_.blobs.length, this.constructor.PAGE_SIZE_));
|
|
this.resetData_();
|
|
this.loadMoreResults();
|
|
};
|
|
|
|
cam.SearchSession.prototype.close = function() {
|
|
if (this.socket_) {
|
|
this.socket_.onerror = null;
|
|
this.socket_.onclose = null;
|
|
this.socket_.close();
|
|
}
|
|
};
|
|
|
|
cam.SearchSession.prototype.getMeta = function(blobref) {
|
|
return this.data_.description.meta[blobref];
|
|
};
|
|
|
|
cam.SearchSession.prototype.getResolvedMeta = function(blobref) {
|
|
var meta = this.data_.description.meta[blobref];
|
|
if (meta && meta.camliType == 'permanode') {
|
|
var camliContent = cam.permanodeUtils.getSingleAttr(meta.permanode, 'camliContent');
|
|
if (camliContent) {
|
|
return this.data_.description.meta[camliContent];
|
|
}
|
|
}
|
|
return meta;
|
|
};
|
|
|
|
cam.SearchSession.prototype.getTitle = function(blobref) {
|
|
var meta = this.getMeta(blobref);
|
|
if (meta.camliType == 'permanode') {
|
|
var title = cam.permanodeUtils.getSingleAttr(meta.permanode, 'title');
|
|
if (title) {
|
|
return title;
|
|
}
|
|
}
|
|
var rm = this.getResolvedMeta(blobref);
|
|
return (rm && rm.camliType == 'file' && rm.file.fileName) || (rm && rm.camliType == 'directory' && rm.dir.fileName) || '';
|
|
};
|
|
|
|
cam.SearchSession.prototype.resetData_ = function() {
|
|
this.data_ = {
|
|
blobs: [],
|
|
description: {
|
|
meta: {}
|
|
}
|
|
};
|
|
};
|
|
|
|
cam.SearchSession.prototype.initSocketUri_ = function(currentUri) {
|
|
if (!goog.global.WebSocket) {
|
|
return;
|
|
}
|
|
|
|
this.socketUri_ = currentUri;
|
|
this.socketUri_.setFragment('');
|
|
var config = this.connection_.getConfig();
|
|
this.socketUri_.setPath(goog.uri.utils.appendPath(config.searchRoot, 'camli/search/ws'));
|
|
this.socketUri_.setQuery(goog.Uri.QueryData.createFromMap({authtoken: config.authToken || ''}));
|
|
if (this.socketUri_.getScheme() == "https") {
|
|
this.socketUri_.setScheme("wss");
|
|
} else {
|
|
this.socketUri_.setScheme("ws");
|
|
}
|
|
};
|
|
|
|
cam.SearchSession.prototype.getContinuation_ = function(changeType, opt_continuationToken, opt_limit) {
|
|
return this.connection_.search.bind(this.connection_, this.query_, cam.ServerConnection.DESCRIBE_REQUEST, opt_limit || this.constructor.PAGE_SIZE_, opt_continuationToken,
|
|
this.searchDone_.bind(this, changeType));
|
|
};
|
|
|
|
cam.SearchSession.prototype.searchDone_ = function(changeType, result) {
|
|
if (!result) {
|
|
result = {};
|
|
}
|
|
if (!result.blobs) {
|
|
result.blobs = [];
|
|
}
|
|
if (!result.description) {
|
|
result.description = {};
|
|
}
|
|
|
|
var changes = false;
|
|
|
|
if (changeType == this.constructor.SEARCH_SESSION_CHANGE_TYPE.APPEND) {
|
|
changes = Boolean(result.blobs.length);
|
|
this.data_.blobs = this.data_.blobs.concat(result.blobs);
|
|
goog.mixin(this.data_.description.meta, result.description.meta);
|
|
} else {
|
|
changes = true;
|
|
this.data_.blobs = result.blobs;
|
|
this.data_.description = result.description;
|
|
}
|
|
|
|
if (result.continue) {
|
|
this.continuation_ = this.getContinuation_(this.constructor.SEARCH_SESSION_CHANGE_TYPE.APPEND, result.continue);
|
|
} else {
|
|
this.continuation_ = null;
|
|
this.isComplete_ = true;
|
|
}
|
|
|
|
if (changes) {
|
|
this.dispatchEvent({type: this.constructor.SEARCH_SESSION_CHANGED, changeType: changeType});
|
|
|
|
if (changeType == this.constructor.SEARCH_SESSION_CHANGE_TYPE.NEW || changeType == this.constructor.SEARCH_SESSION_CHANGE_TYPE.APPEND) {
|
|
this.startSocketQuery_();
|
|
}
|
|
}
|
|
};
|
|
|
|
cam.SearchSession.prototype.handleError_ = function(message) {
|
|
this.hasSocketError_ = true;
|
|
this.dispatchEvent({type: this.constructor.SEARCH_SESSION_ERROR});
|
|
};
|
|
|
|
cam.SearchSession.prototype.handleStatus_ = function(data) {
|
|
if (data.tag == '_status') {
|
|
this.dispatchEvent({
|
|
type: this.constructor.SEARCH_SESSION_STATUS,
|
|
status: data.status,
|
|
});
|
|
}
|
|
};
|
|
|
|
cam.SearchSession.prototype.startSocketQuery_ = function() {
|
|
if (!this.socketUri_) {
|
|
return;
|
|
}
|
|
|
|
this.close();
|
|
|
|
var numResults = 0;
|
|
if (this.data_ && this.data_.blobs) {
|
|
numResults = this.data_.blobs.length;
|
|
}
|
|
var query = this.connection_.buildQuery(this.query_, cam.ServerConnection.DESCRIBE_REQUEST, Math.max(numResults, this.constructor.PAGE_SIZE_), null, this.around_);
|
|
|
|
this.socket_ = new WebSocket(this.socketUri_.toString());
|
|
this.socket_.onopen = function() {
|
|
var message = {
|
|
tag: this.tag_,
|
|
query: query
|
|
};
|
|
this.socket_.send(JSON.stringify(message));
|
|
}.bind(this);
|
|
this.socket_.onclose =
|
|
this.socket_.onerror = function(e) {
|
|
this.handleError_('WebSocket error - click to reload');
|
|
}.bind(this);
|
|
this.socket_.onmessage = function(e) {
|
|
this.supportsWebSocket_ = true;
|
|
this.handleStatus_(JSON.parse(e.data));
|
|
// Ignore the first response.
|
|
this.socket_.onmessage = function(e) {
|
|
var result = JSON.parse(e.data);
|
|
this.handleStatus_(result);
|
|
if (result.tag == this.tag_) {
|
|
this.searchDone_(this.constructor.SEARCH_SESSION_CHANGE_TYPE.UPDATE, result.result);
|
|
}
|
|
}.bind(this);
|
|
}.bind(this);
|
|
};
|