@harishreddym/baqend
Version:
Baqend JavaScript SDK
747 lines (564 loc) • 31.3 kB
HTML
<html lang="en">
<head>
<meta charset="utf-8">
<title>Baqend JavaScript SDK 2.14.1 - Source: realtime/query/Stream.js</title>
<script src="scripts/prettify/prettify.js"> </script>
<script src="scripts/prettify/lang-css.js"> </script>
<!--[if lt IE 9]>
<script src="//html5shiv.googlecode.com/svn/trunk/html5.js"></script>
<![endif]-->
<link rel="shortcut icon" type="image/x-icon" href="img/favicon.ico">
<link type="text/css" rel="stylesheet" href="styles/prettify-tomorrow.css">
<link type="text/css" rel="stylesheet" href="styles/bootstrap-baqend.min.css">
<link type="text/css" rel="stylesheet" href="styles/jsdoc-default.css">
<link type="text/css" rel="stylesheet" href="styles/font-awesome-4.7.0.css">
</head>
<body>
<nav class="navbar navbar-default navbar-fixed-top" role="navigation">
<div class="navbar-inner container">
<!-- Collapsed navigation -->
<div class="navbar-header">
<!-- Expander button -->
<button type="button" class="navbar-toggle" data-toggle="collapse" data-target=".navbar-collapse">
<span class="sr-only">Toggle navigation</span>
<span class="icon-bar"></span>
<span class="icon-bar"></span>
<span class="icon-bar"></span>
</button>
<!-- Main title -->
<a class="navbar-brand" href="/"><img src="img/logo.png"></a>
</div>
<!-- Expanded navigation -->
<div id="nav" class="navbar-collapse collapse">
<!-- Search -->
<form class="navbar-right form-inline search-form">
<div class="form-group search-form-group">
<input type="search" class="form-control search-input" id="search-query" placeholder="Search Guide" name="q" autocomplete="off">
<div id="search-results" class="search-results">
<p class="search-no-results">Please enter a search query ...</p>
</div>
<i class="search-icon fa fa-search"></i>
</div>
</form>
<!-- Main navigation -->
<ul class="nav navbar-nav">
<li>
<a href="https://dashboard.baqend.com/">Dashboard</a>
</li>
<!-- Guide Navigation -->
<li class="dropdown">
<a href="#" class="dropdown-toggle" data-toggle="dropdown">Guide <b class="caret"></b></a>
<ul class="dropdown-menu">
<li><a href="https://www.baqend.com/guide/">Home</a></li>
<li><a href="https://www.baqend.com/guide/#speed-kit">Speed Kit</a></li>
<li><a href="https://www.baqend.com/guide/#platform">Platform</a></li>
<li><a href="https://www.baqend.com/guide/roadmap/">Roadmap</a></li>
<li><a href="https://www.baqend.com/guide/topics/faq/">FAQ</a></li>
</ul>
</li>
<!-- Starter Kits -->
<li class="dropdown">
<a href="#" class="dropdown-toggle" data-toggle="dropdown">Starter Kits <b class="caret"></b></a>
<ul class="dropdown-menu">
<li><a href="https://www.baqend.com/guide/starter-kits/">Starter Kits Overview</a></li>
<li><a href="https://www.baqend.com/guide/starter-kits/angular2/">Angular 2 Starter Kit</a></li>
<li><a href="https://www.baqend.com/guide/starter-kits/react/">React and Redux Starter Kit</a></li>
<li><a href="https://www.baqend.com/guide/starter-kits/bootstrap/">Bootstrap Starter Kit</a></li>
<li><a href="https://www.baqend.com/guide/starter-kits/ionic2/">Ionic 2 Starter Kit</a></li>
<li><a href="https://www.baqend.com/guide/starter-kits/ionic/">Ionic Starter Kit</a></li>
</ul>
</li>
<li class="active">
<a href="baqend.html">JS API</a>
</li>
<li>
<a href="https://www.baqend.com/tutorial.html">Tutorial</a>
</li>
</ul>
</div>
</div>
</nav>
<div class="box gray pt-32 pb-32">
<div class="container">
<div class="row">
<!-- <div class="col-md-3"></div> -->
<div class="bs-sidebar hidden-print fixed affix" role="complementary">
<input class="filter form-control input-sm" type="text" placeholder="FILTER"/>
<ul class="nav bs-sidenav">
<li class=""><a href="Acl.html" class="nav-name">Acl</a></li>
<li class=""><a href="EntityManager.html" class="nav-name">EntityManager</a></li>
<li class=""><a href="EntityManagerFactory.html" class="nav-name">EntityManagerFactory</a></li>
<li class=""><a href="GeoPoint.html" class="nav-name">GeoPoint</a></li>
<li class=""><a href="RealtimeEvent.html" class="nav-name">RealtimeEvent</a></li>
<li class=""><a href="baqend.html" class="nav-name">baqend</a></li>
<li class="">
<a href="binding.html" class="nav-name">binding</a>
<ul class="nav">
<li class=""><a href="binding.Accessor.html" class="nav-name">Accessor</a></li>
<li class=""><a href="binding.DeviceFactory.html" class="nav-name">DeviceFactory</a></li>
<li class=""><a href="binding.Enhancer.html" class="nav-name">Enhancer</a></li>
<li class=""><a href="binding.Entity.html" class="nav-name">Entity</a></li>
<li class=""><a href="binding.EntityFactory.html" class="nav-name">EntityFactory</a></li>
<li class=""><a href="binding.Factory.html" class="nav-name">Factory</a></li>
<li class=""><a href="binding.File.html" class="nav-name">File</a></li>
<li class=""><a href="binding.FileFactory.html" class="nav-name">FileFactory</a></li>
<li class=""><a href="binding.Managed.html" class="nav-name">Managed</a></li>
<li class=""><a href="binding.ManagedFactory.html" class="nav-name">ManagedFactory</a></li>
<li class=""><a href="binding.Role.html" class="nav-name">Role</a></li>
<li class=""><a href="binding.User.html" class="nav-name">User</a></li>
<li class=""><a href="binding.UserFactory.html" class="nav-name">UserFactory</a></li>
</ul>
</li>
<li class="">
<a href="caching.html" class="nav-name">caching</a>
<ul class="nav">
<li class=""><a href="caching.BloomFilter.html" class="nav-name">BloomFilter</a></li>
</ul>
</li>
<li class="">
<a href="connector.html" class="nav-name">connector</a>
<ul class="nav">
<li class=""><a href="connector.Connector.html" class="nav-name">Connector</a></li>
<li class=""><a href="connector.FetchConnector.html" class="nav-name">FetchConnector</a></li>
<li class=""><a href="connector.IFrameConnector.html" class="nav-name">IFrameConnector</a></li>
<li class=""><a href="connector.Message.html" class="nav-name">Message</a></li>
<li class=""><a href="connector.NodeConnector.html" class="nav-name">NodeConnector</a></li>
<li class=""><a href="connector.ObservableStream.html" class="nav-name">ObservableStream</a></li>
<li class=""><a href="connector.WebSocketConnector.html" class="nav-name">WebSocketConnector</a></li>
<li class=""><a href="connector.XMLHttpConnector.html" class="nav-name">XMLHttpConnector</a></li>
<li class=""><a href="connector.ChannelMessage.html" class="nav-name">ChannelMessage</a></li>
</ul>
</li>
<li class="">
<a href="error.html" class="nav-name">error</a>
<ul class="nav">
<li class=""><a href="error.CommunicationError.html" class="nav-name">CommunicationError</a></li>
<li class=""><a href="error.EntityExistsError.html" class="nav-name">EntityExistsError</a></li>
<li class=""><a href="error.IllegalEntityError.html" class="nav-name">IllegalEntityError</a></li>
<li class=""><a href="error.PersistentError.html" class="nav-name">PersistentError</a></li>
<li class=""><a href="error.RollbackError.html" class="nav-name">RollbackError</a></li>
</ul>
</li>
<li class="">
<a href="metamodel.html" class="nav-name">metamodel</a>
<ul class="nav">
<li class=""><a href="metamodel.Attribute.html" class="nav-name">Attribute</a></li>
<li class=""><a href="metamodel.BasicType.html" class="nav-name">BasicType</a></li>
<li class=""><a href="metamodel.CollectionAttribute.html" class="nav-name">CollectionAttribute</a></li>
<li class=""><a href="metamodel.DbIndex.html" class="nav-name">DbIndex</a></li>
<li class=""><a href="metamodel.EmbeddableType.html" class="nav-name">EmbeddableType</a></li>
<li class=""><a href="metamodel.EntityType.html" class="nav-name">EntityType</a></li>
<li class=""><a href="metamodel.ListAttribute.html" class="nav-name">ListAttribute</a></li>
<li class=""><a href="metamodel.ManagedType.html" class="nav-name">ManagedType</a></li>
<li class=""><a href="metamodel.MapAttribute.html" class="nav-name">MapAttribute</a></li>
<li class=""><a href="metamodel.Metamodel.html" class="nav-name">Metamodel</a></li>
<li class=""><a href="metamodel.ModelBuilder.html" class="nav-name">ModelBuilder</a></li>
<li class=""><a href="metamodel.PluralAttribute.html" class="nav-name">PluralAttribute</a></li>
<li class=""><a href="metamodel.SetAttribute.html" class="nav-name">SetAttribute</a></li>
<li class=""><a href="metamodel.SingularAttribute.html" class="nav-name">SingularAttribute</a></li>
<li class=""><a href="metamodel.Type.html" class="nav-name">Type</a></li>
</ul>
</li>
<li class="">
<a href="model.html" class="nav-name">model</a>
<ul class="nav">
<li class=""><a href="model.Device.html" class="nav-name">Device</a></li>
<li class=""><a href="model.Role.html" class="nav-name">Role</a></li>
<li class=""><a href="model.User.html" class="nav-name">User</a></li>
</ul>
</li>
<li class="">
<a href="partialupdate.html" class="nav-name">partialupdate</a>
<ul class="nav">
<li class=""><a href="partialupdate.EntityPartialUpdateBuilder.html" class="nav-name">EntityPartialUpdateBuilder</a></li>
<li class=""><a href="partialupdate.PartialUpdateBuilder.html" class="nav-name">PartialUpdateBuilder</a></li>
<li class=""><a href="partialupdate.UpdateOperation.html" class="nav-name">UpdateOperation</a></li>
</ul>
</li>
<li class="">
<a href="query.html" class="nav-name">query</a>
<ul class="nav">
<li class=""><a href="query.Builder.html" class="nav-name">Builder</a></li>
<li class=""><a href="query.Filter.html" class="nav-name">Filter</a></li>
<li class=""><a href="query.Node.html" class="nav-name">Node</a></li>
<li class=""><a href="query.Operator.html" class="nav-name">Operator</a></li>
<li class=""><a href="query.Query.html" class="nav-name">Query</a></li>
<li class=""><a href="query.Stream.html" class="nav-name">Stream</a></li>
<li class=""><a href="query.Condition.html" class="nav-name">Condition</a></li>
</ul>
</li>
<li class="">
<a href="util.html" class="nav-name">util</a>
<ul class="nav">
<li class=""><a href="util.Code.html" class="nav-name">Code</a></li>
<li class=""><a href="util.Lockable.html" class="nav-name">Lockable</a></li>
<li class=""><a href="util.Logger.html" class="nav-name">Logger</a></li>
<li class=""><a href="util.Metadata.html" class="nav-name">Metadata</a></li>
<li class=""><a href="util.Modules.html" class="nav-name">Modules</a></li>
<li class=""><a href="util.Permission.html" class="nav-name">Permission</a></li>
<li class=""><a href="util.PushMessage.html" class="nav-name">PushMessage</a></li>
<li class=""><a href="util.TokenStorage.html" class="nav-name">TokenStorage</a></li>
<li class=""><a href="util.ValidationResult.html" class="nav-name">ValidationResult</a></li>
<li class=""><a href="util.Validator.html" class="nav-name">Validator</a></li>
<li class=""><a href="util.TokenStorageFactory.html" class="nav-name">TokenStorageFactory</a></li>
</ul>
</li>
</ul>
</div>
<div class="col-md-12" id="main">
<div class="content">
<h1 class="page-title">Source: realtime/query/Stream.js</h1>
<section>
<article>
<pre class="prettyprint source linenums"><code>'use strict';
const Metadata = require('../../lib/util/Metadata');
const lib = require('../../lib/baqend');
const uuid = require('../../lib/util/uuid').uuid;
/**
* @typedef {object} StreamOptions
* @property {boolean} initial Indicates whether or not the initial result set should be delivered on
* creating the subscription.
* @property {Array<string>} matchTypes A list of match types.
* @property {Array<string>} operations A list of operations.
* @property {number} reconnects The number of reconnects.
*/
/**
* @alias query.Stream
*/
class Stream {
/**
* Creates a live updating object stream for a query
*
* @alias query.Stream.createStream<T>
* @param {EntityManager} entityManager The owning entity manager of this query
* @param {string} query The query options
* @param {string} query.query The serialized query
* @param {string} query.bucket The Bucket on which the streaming query is performed
* @param {string=} query.sort the sort string
* @param {number=} query.limit the count, i.e. the number of items in the result
* @param {number=} query.offset offset, i.e. the number of items to skip
* @param {boolean=} query.initial Indicates if the initial result should be returned
* @param {Partial<StreamOptions>} options an object containing parameters
* @return {Observable<RealtimeEvent<T>>} The query result as a live updating stream of objects
*/
static createEventStream(entityManager, query, options) {
const opt = options || {};
opt.reconnects = 0;
return Stream.streamObservable(entityManager, query, opt, (msg, next) => {
const messageType = msg.type;
delete msg.type;
if (messageType === 'result') {
msg.data.forEach((obj, index) => {
const event = Object.assign({
matchType: 'add',
operation: 'none',
initial: true,
}, msg);
event.data = Stream.resolveObject(entityManager, obj);
if (query.sort) { event.index = index; }
next(event);
});
}
if (messageType === 'match') {
msg.data = Stream.resolveObject(entityManager, msg.data);
next(msg);
}
});
}
/**
* Creates a live updating result stream for a query
*
* @alias query.Stream.createStreamResult<T>
* @param {EntityManager} entityManager The owning entity manager of this query
* @param {string} query The query options
* @param {string} query.query The serialized query
* @param {string} query.bucket The Bucket on which the streaming query is performed
* @param {string=} query.sort the sort string
* @param {number=} query.limit the count, i.e. the number of items in the result
* @param {number=} query.offset offset, i.e. the number of items to skip
* @param {Partial<StreamOptions>} options an object containing parameters
* @return {Observable<Array<T>>} The query result as a live updating query result
*/
static createResultStream(entityManager, query, options) {
const opt = options || {};
opt.initial = true;
opt.matchTypes = 'all';
opt.operations = 'any';
let result;
const ordered = !!query.sort;
return Stream.streamObservable(entityManager, query, opt, (event, next) => {
if (event.type === 'result') {
result = event.data.map(obj => Stream.resolveObject(entityManager, obj));
next(result.slice());
}
if (event.type === 'match') {
const obj = Stream.resolveObject(entityManager, event.data);
if (event.matchType === 'remove' || event.matchType === 'changeIndex') {
// if we have removed the instance our self, we do not have the cached instances anymore
// therefore we can't find it anymore in the result by identity
for (let i = 0, len = result.length; i < len; i += 1) {
if (result[i].id === event.data.id) {
result.splice(i, 1);
break;
}
}
}
if (event.matchType === 'add' || event.matchType === 'changeIndex') {
if (ordered) {
result.splice(event.index, 0, obj);
} else {
result.push(obj);
}
}
next(result.slice());
}
});
}
static streamObservable(entityManager, query, options, mapper) {
const opt = Stream.parseOptions(options);
const socket = entityManager.entityManagerFactory.websocket;
const observable = new lib.Observable((subscriber) => {
const id = uuid();
const stream = socket.openStream(entityManager.tokenStorage, id);
stream.send(Object.assign({
type: 'subscribe',
}, query, opt));
let closed = false;
const next = subscriber.next.bind(subscriber);
const subscription = stream.subscribe({
complete() {
closed = true;
subscriber.complete();
},
error(e) {
closed = true;
subscriber.error(e);
},
next(msg) {
mapper(msg, next);
},
});
return () => {
if (!closed) { // send unsubscribe only when we aren't completed by the socket and call it only once
stream.send({ type: 'unsubscribe' });
subscription.unsubscribe();
closed = true;
}
};
});
return Stream.cachedObservable(observable, opt);
}
static cachedObservable(observable, options) {
let subscription = null;
const observers = [];
return new lib.Observable((observer) => {
if (!subscription) {
let remainingRetries = options.reconnects;
let backoff = 1;
const subscriptionObserver = {
next(msg) {
// reset the backoff if we get a message
backoff = 1;
observers.forEach(o => o.next(msg));
},
error(e) {
observers.forEach(o => o.error(e));
},
complete() {
if (remainingRetries !== 0) {
remainingRetries = remainingRetries < 0 ? -1 : remainingRetries - 1;
setTimeout(() => {
subscription = observable.subscribe(subscriptionObserver);
}, backoff * 1000);
backoff *= 2;
} else {
observers.forEach(o => o.complete());
}
},
};
subscription = observable.subscribe(subscriptionObserver);
}
observers.push(observer);
return () => {
observers.splice(observers.indexOf(observer), 1);
if (!observers.length) {
subscription.unsubscribe();
subscription = null;
}
};
});
}
/**
* Parses the StreamOptions
*
* @param {Partial<StreamOptions>=} [options] object containing partial options
* @returns {StreamOptions} an object containing VALID options
*/
static parseOptions(options) {
const opt = options || {};
const verified = {
initial: opt.initial === undefined || !!opt.initial,
matchTypes: Stream.normalizeMatchTypes(opt.matchTypes),
operations: Stream.normalizeOperations(opt.operations),
reconnects: Stream.normalizeReconnects(opt.reconnects),
};
if (verified.matchTypes.indexOf('all') === -1 && verified.operations.indexOf('any') === -1) {
throw new Error('Only subscriptions for either operations or matchTypes are allowed. You cannot subscribe to a query using matchTypes and operations at the same time!');
}
return verified;
}
static normalizeMatchTypes(list) {
return Stream.normalizeSortedSet(list, 'all', 'match types', ['add', 'change', 'changeIndex', 'match', 'remove']);
}
static normalizeReconnects(reconnects) {
if (reconnects === undefined) {
return -1;
}
return reconnects < 0 ? -1 : Number(reconnects);
}
static normalizeOperations(list) {
return Stream.normalizeSortedSet(list, 'any', 'operations', ['delete', 'insert', 'none', 'update']);
}
static normalizeSortedSet(list, wildcard, itemType, allowedItems) {
if (!list) {
return [wildcard];
}
const li = Array.isArray(list) ? list : [list];
if (li.length === 0) { // undefined or empty list --> default value
return [wildcard];
}
// sort, remove duplicates and check whether all values are allowed
li.sort();
let item;
let lastItem;
for (let i = li.length - 1; i >= 0; i -= 1) {
item = li[i];
if (!item) { // undefined and null item in the list --> invalid!
throw new Error('undefined and null not allowed!');
}
if (item === lastItem) { // remove duplicates
li.splice(i, 1);
}
if (item === wildcard) {
return [wildcard];
}
if (allowedItems.indexOf(item) === -1) { // raise error on invalid elements
throw new Error(item + ' not allowed for ' + itemType + '! (permitted: ' + allowedItems + '.)');
}
lastItem = item;
}
return li;
}
static resolveObject(entityManager, object) {
const entity = entityManager.getReference(object.id);
const metadata = Metadata.get(entity);
if (!object.version) {
metadata.setRemoved();
entityManager.removeReference(entity);
} else if (entity.version <= object.version) {
metadata.setJson(object, { persisting: true });
}
return entity;
}
}
module.exports = Stream;
</code></pre>
</article>
</section>
<!-- <footer>
<strong>Baqend JavaScript SDK 2.14.1</strong><br>
© 2019 Baqend GmbH<br>
Documentation generated by <a href="https://github.com/jsdoc3/jsdoc">JSDoc 3.5.5</a> on Sun, 06 Jan 2019 05:00:21 GMT
</footer> -->
</div>
</div>
</div>
</div>
</div>
<div id="footer" style="position:relative; z-index:1;">
<div class="container">
<div class="row">
<div class="col-xs-6 col-sm-3 sitemap">
<h4>Product</h4>
<a href="https://www.baqend.com/platform.html">Platform</a>
<a href="https://www.baqend.com/speedkit.html">Speed Kit</a>
<a href="https://www.baqend.com/features.html">Features</a>
<a href="https://www.baqend.com/pricing.html">Platform Pricing</a>
<a href="https://www.baqend.com/pricing_speedkit.html">Speed Kit Pricing</a>
<a href="https://www.baqend.com/agencies.html">Agencies</a>
<a href="https://www.baqend.com/enterprise.html">Enterprise</a>
<a href="https://www.baqend.com/features.html#download">Community Edition</a>
<a href="https://www.baqend.com/support.html">Support</a>
</div>
<div class="col-xs-6 col-sm-3 sitemap">
<h4>Company</h4>
<a href="https://www.baqend.com/about.html">About us</a>
<a href="https://medium.baqend.com">Baqend Blog</a>
<a href="https://www.baqend.com/hiring.html">Jobs</a>
<a href="https://www.baqend.com/press.html">Press</a>
<a href="https://thesis.app.baqend.com/">Bachelor/Master theses</a>
<a href="https://www.weblabs.hamburg/">WebLabs.Hamburg</a>
</div>
<div class="col-xs-6 col-sm-3 sitemap">
<h4>Developer</h4>
<a href="https://dashboard.baqend.com/login">Login</a>
<a href="https://dashboard.baqend.com/register">Sign Up</a>
<a href="https://www.baqend.com/guide">Developer Docs</a>
<a href="https://www.baqend.com/guide/roadmap/">Roadmap</a>
<a href="https://www.baqend.com/guide/starter-kits">Starter Kits</a>
<a href="https://www.baqend.com/js-sdk/latest/baqend.html">JS SDK</a>
<a href="https://stackoverflow.com/questions/tagged/baqend/">Stackoverflow</a>
</div>
<div class="col-xs-6 col-sm-3 contact">
<h4>Contact</h4>
<p class="contact-information">
<span>
Baqend GmbH<br />
Stresemannstr. 23<br />
22769 Hamburg<br />
Germany
</span>
</p>
<p class="contact-information">
Email: <a style="display: inline-block" href="mailto:support@baqend.com">support@baqend.com</a>
</p>
<p class="contact-information">
Tel: <a style="display: inline-block" href="tel:+494060940539">+49 40 60940539</a>
</p>
</div>
</div>
<div class="row">
<div class="col-md-12 text-center social">
<a href="https://twitter.com/baqendcom"><i class="fa fa-twitter fa-fw"></i></a>
<a href="https://github.com/Baqend"><i class="fa fa-github fa-fw"></i></a>
<a href="https://medium.baqend.com/"><i class="fa fa-medium fa-fw"></i></a>
<a href="https://www.facebook.com/baqend"><i class="fa fa-facebook fa-fw"></i></a>
<a href="https://www.youtube.com/channel/UCsImg6Ts8UEp6-7LE9CP2-Q"><i class="fa fa-youtube fa-fw"></i></a>
</div>
</div>
</div>
<footer>
<div class="container">
<div class="legal text-center">
<div>
<strong>Baqend JavaScript SDK 2.14.1</strong><br>
© 2019 Baqend GmbH<br>
Documentation generated by <a href="https://github.com/jsdoc3/jsdoc">JSDoc 3.5.5</a> on Sun, 06 Jan 2019 05:00:21 GMT
</div>
<a href="mailto:info@baqend.com">Contact</a> -
<a href="https://dashboard.baqend.com/privacy">Privacy Policy</a> -
<a href="https://dashboard.baqend.com/terms">Terms of Service</a> -
<a href="https://dashboard.baqend.com/imprint">Imprint</a>
</div>
</div>
</footer>
</div>
<script> prettyPrint(); </script>
<script src="https://code.jquery.com/jquery-1.12.1.min.js"></script>
<script src="scripts/bootstrap.min.js"> </script>
<script src="scripts/linenumber.js"> </script>
<script src="scripts/filter.js"> </script>
<script data-main="scripts/search.js" src="scripts/require.js"></script>
</body>
</html>