forked from cometd/cometd-javascript
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathReloadExtension.js
233 lines (210 loc) · 9.6 KB
/
ReloadExtension.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
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
/*
* Copyright (c) 2008-2022 the original author or authors.
*
* 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.
*/
(((root, factory) => {
if (typeof exports === 'object') {
module.exports = factory(require('./cometd'));
} else if (typeof define === 'function' && define.amd) {
define(['./cometd'], factory);
} else {
factory(root.org.cometd);
}
})(this, cometdModule => {
/**
* The reload extension allows a page to be loaded (or reloaded)
* without having to re-handshake in the new (or reloaded) page,
* therefore resuming the existing CometD connection.
*
* When the reload() method is called, the state of the CometD
* connection is stored in the window.sessionStorage object.
* The reload() method must therefore be called by page unload
* handlers, often provided by JavaScript toolkits.
*
* When the page is (re)loaded, this extension checks the
* window.sessionStorage and restores the CometD connection,
* maintaining the same CometD clientId.
*/
return cometdModule.ReloadExtension = function(configuration) {
let _cometd;
let _debug;
let _state = {};
let _name = 'org.cometd.reload';
let _batch = false;
let _reloading = false;
function _reload(config) {
if (_state.handshakeResponse) {
_reloading = true;
const transport = _cometd.getTransport();
if (transport) {
transport.abort();
}
_configure(config);
const state = JSON.stringify(_state);
_debug('Reload extension saving state', state);
window.sessionStorage.setItem(_name, state);
}
}
function _similarState(oldState) {
// We want to check here that the CometD object
// did not change much between reloads.
// We just check the URL for now, but in future
// further checks may involve the transport type
// and other configuration parameters.
return _state.url === oldState.url;
}
function _configure(config) {
if (config) {
if (typeof config.name === 'string') {
_name = config.name;
}
}
}
function _receive(response) {
_cometd.receive(response);
}
this.configure = _configure;
this._receive = _receive;
this.registered = (name, cometd) => {
_cometd = cometd;
_cometd.reload = _reload;
_debug = _cometd._debug;
};
this.unregistered = () => {
delete _cometd.reload;
_cometd = null;
};
this.outgoing = function(message) {
switch (message.channel) {
case '/meta/handshake': {
_state = {};
_state.url = _cometd.getURL();
const state = window.sessionStorage.getItem(_name);
_debug('Reload extension found state', state);
// Is there a saved handshake response from a prior load ?
if (state) {
try {
const oldState = JSON.parse(state);
// Remove the state, not needed anymore
window.sessionStorage.removeItem(_name);
if (oldState.handshakeResponse && _similarState(oldState)) {
_debug('Reload extension restoring state', oldState);
// Since we are going to abort this message,
// we must save an eventual callback to restore
// it when we replay the handshake response.
const callback = _cometd._getCallback(message.id);
setTimeout(() => {
_debug('Reload extension replaying handshake response', oldState.handshakeResponse);
_state.handshakeResponse = oldState.handshakeResponse;
_state.transportType = oldState.transportType;
// Restore the callback.
_cometd._putCallback(message.id, callback);
const response = _cometd._mixin(true, {}, _state.handshakeResponse, {
// Keep the response message id the same as the request.
id: message.id,
// Tells applications this is a handshake replayed by the reload extension.
ext: {
reload: true
}
});
// Use the same transport as before.
response.supportedConnectionTypes = [_state.transportType];
this._receive(response);
_debug('Reload extension replayed handshake response', response);
}, 0);
// Delay any sends until first connect is complete.
// This avoids that there is an old /meta/connect pending on server
// that will be resumed to send messages to the client, when the
// client has already closed the connection, thereby losing the messages.
if (!_batch) {
_batch = true;
_cometd.startBatch();
}
// This handshake is aborted, as we will replay the prior handshake response
return null;
} else {
_debug('Reload extension could not restore state', oldState);
}
} catch (x) {
_debug('Reload extension error while trying to restore state', x);
}
}
break;
}
case '/meta/connect': {
if (_reloading === true) {
// The reload causes the failure of the outstanding /meta/connect,
// which CometD will react to by sending another. Here we avoid
// that /meta/connect messages are sent between the reload and
// the destruction of the JavaScript context, so that we are sure
// that the first /meta/connect is the one triggered after the
// replay of the /meta/handshake by this extension.
_debug('Reload extension aborting /meta/connect during reload');
return null;
}
if (!_state.transportType) {
_state.transportType = message.connectionType;
_debug('Reload extension tracked transport type', _state.transportType);
}
break;
}
case '/meta/disconnect': {
_state = {};
break;
}
default: {
break;
}
}
return message;
};
this.incoming = message => {
switch (message.channel) {
case '/meta/handshake': {
// Only record the handshake response if it's successful.
if (message.successful) {
// If the handshake response is already present, then we're replaying it.
// Since the replay may have modified the handshake response, do not record it again.
if (!_state.handshakeResponse) {
// Save successful handshake response
_state.handshakeResponse = message;
_debug('Reload extension tracked handshake response', message);
}
}
break;
}
case '/meta/connect': {
if (_batch) {
_batch = false;
_cometd.endBatch();
}
break;
}
case '/meta/disconnect': {
if (_batch) {
_batch = false;
_cometd.endBatch();
}
_state = {};
break;
}
default: {
break;
}
}
return message;
};
_configure(configuration);
};
}));