-
Notifications
You must be signed in to change notification settings - Fork 49
/
Copy pathSession.js
85 lines (79 loc) · 3.75 KB
/
Session.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
import {ensure} from './util';
import {SubscriptionManager} from './SubscriptionManager';
// A Session encapsulates a RethinkDB connection and active queries that React
// components are currently subscribed to. You can have multiple independent
// Sessions (useful for isomorphic apps), or you can use the DefaultSession
// singleton for convenience. Each rethink mixin must be associated with a
// Session. The DefaultMixin singleton is associated with DefaultSession.
//
// Call connect() to open a RethinkDB websocket connection, and close() to
// disconnect. Call runQuery() to run a RethinkDB query on the connection,
// returning a Promise with the results. You must call connect() before calling
// runQuery() or mounting a component using a mixin for this session. You do
// not have to wait for it to finish connecting, as it will automatically wait
// on the connection promise.
//
// Call onceDoneLoading() with a callback to be notified when there are no
// active queries waiting on results.
//
// You may reopen a closed session. A typical use case is to have one session
// per application (likely DefaultSession), to call connect() when the user
// authenticates (including authentication information in the path), and
// close() when the user signs out.
//
// The exported MetaSession function allows dependency injection of
// RethinkdbWebsocketClient, so that we can the resulting Session class work in
// either node.js or the browser.
export const MetaSession = RethinkdbWebsocketClient => {
const {Promise, connect} = RethinkdbWebsocketClient;
return class Session {
constructor() {
const runQueryFn = this.runQuery.bind(this);
this._connPromise = null;
this._subscriptionManager = new SubscriptionManager(runQueryFn);
}
connect({host, port, path, wsProtocols, wsBinaryType, secure, db, simulatedLatencyMs, autoReconnectDelayMs}) {
ensure(!this._connPromise, 'Session.connect() called when connected');
const connectAfterDelay = delayMs => {
const options = {host, port, path, wsProtocols, wsBinaryType, secure, db, simulatedLatencyMs};
this._connPromise = new Promise((resolve, reject) => {
setTimeout(() => {
connect(options).then(resolve, reject);
}, delayMs);
});
const onClose = () => {
this._subscriptionManager.handleDisconnect();
// Don't trigger on client initiated Session.close()
if (this._connPromise) {
console.warn('RethinkDB WebSocket connection failure.',
`Reconnecting in ${autoReconnectDelayMs}ms`);
connectAfterDelay(autoReconnectDelayMs);
}
};
if (autoReconnectDelayMs !== undefined) {
this._connPromise.then(conn => conn.on('close', onClose), onClose);
}
this._connPromise.then(() => this._subscriptionManager.handleConnect());
};
connectAfterDelay(0);
}
close() {
ensure(this._connPromise, 'Session.close() called when not connected');
this._connPromise.then(conn => conn.close());
this._connPromise = null;
}
runQuery(query, options) {
ensure(this._connPromise, 'Must connect() before calling runQuery()');
// Rather than calling query.run(c), we create a new rethinkdb term and
// use its run function. That way, if the provided query comes from a
// different RethinkdbWebsocketClient, it'll run using the current
// environment. This is mainly to workaround an instanceof check in the
// rethinkdb driver.
const {run} = new RethinkdbWebsocketClient.rethinkdb(null);
return this._connPromise.then(c => run.bind(query)(c, options || {}));
}
onceDoneLoading(callback) {
this._subscriptionManager.onceDoneLoading(callback);
}
};
};