@stomp/stompjs 7.0.0-beta1 → 7.0.0-beta2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (49) hide show
  1. package/bundles/stomp.umd.js +1727 -2043
  2. package/bundles/stomp.umd.js.map +1 -0
  3. package/bundles/stomp.umd.min.js +1 -1
  4. package/esm6/augment-websocket.d.ts +1 -1
  5. package/esm6/augment-websocket.js +1 -1
  6. package/esm6/augment-websocket.js.map +1 -1
  7. package/esm6/client.d.ts +6 -6
  8. package/esm6/client.js +126 -141
  9. package/esm6/client.js.map +1 -1
  10. package/esm6/compatibility/compat-client.d.ts +3 -3
  11. package/esm6/compatibility/compat-client.js +2 -2
  12. package/esm6/compatibility/compat-client.js.map +1 -1
  13. package/esm6/compatibility/heartbeat-info.d.ts +1 -1
  14. package/esm6/compatibility/stomp.d.ts +1 -1
  15. package/esm6/compatibility/stomp.js +2 -2
  16. package/esm6/compatibility/stomp.js.map +1 -1
  17. package/esm6/frame-impl.d.ts +3 -3
  18. package/esm6/frame-impl.js +1 -1
  19. package/esm6/frame-impl.js.map +1 -1
  20. package/esm6/i-frame.d.ts +1 -1
  21. package/esm6/i-message.d.ts +2 -2
  22. package/esm6/index.d.ts +13 -13
  23. package/esm6/index.js +13 -13
  24. package/esm6/index.js.map +1 -1
  25. package/esm6/parser.d.ts +1 -1
  26. package/esm6/stomp-config.d.ts +3 -3
  27. package/esm6/stomp-handler.d.ts +6 -6
  28. package/esm6/stomp-handler.js +6 -6
  29. package/esm6/stomp-handler.js.map +1 -1
  30. package/esm6/stomp-subscription.d.ts +1 -1
  31. package/esm6/types.d.ts +4 -4
  32. package/esm6/types.js +2 -2
  33. package/esm6/types.js.map +1 -1
  34. package/index.d.ts +1 -1
  35. package/package.json +6 -5
  36. package/src/augment-websocket.ts +1 -1
  37. package/src/client.ts +7 -7
  38. package/src/compatibility/compat-client.ts +4 -4
  39. package/src/compatibility/heartbeat-info.ts +1 -1
  40. package/src/compatibility/stomp.ts +3 -3
  41. package/src/frame-impl.ts +4 -4
  42. package/src/i-frame.ts +1 -1
  43. package/src/i-message.ts +2 -2
  44. package/src/index.ts +13 -13
  45. package/src/parser.ts +1 -1
  46. package/src/stomp-config.ts +3 -3
  47. package/src/stomp-handler.ts +11 -11
  48. package/src/stomp-subscription.ts +1 -1
  49. package/src/types.ts +4 -4
@@ -1,2187 +1,1871 @@
1
- (function webpackUniversalModuleDefinition(root, factory) {
2
- if(typeof exports === 'object' && typeof module === 'object')
3
- module.exports = factory();
4
- else if(typeof define === 'function' && define.amd)
5
- define("StompJs", [], factory);
6
- else if(typeof exports === 'object')
7
- exports["StompJs"] = factory();
8
- else
9
- root["StompJs"] = factory();
10
- })(typeof self !== 'undefined' ? self : this, () => {
11
- return /******/ (() => { // webpackBootstrap
12
- /******/ "use strict";
13
- /******/ var __webpack_modules__ = ({
1
+ (function (global, factory) {
2
+ typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) :
3
+ typeof define === 'function' && define.amd ? define(['exports'], factory) :
4
+ (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.StompJs = {}));
5
+ })(this, (function (exports) { 'use strict';
14
6
 
15
- /***/ "./src/augment-websocket.ts":
16
- /*!**********************************!*\
17
- !*** ./src/augment-websocket.ts ***!
18
- \**********************************/
19
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
20
-
21
- __webpack_require__.r(__webpack_exports__);
22
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
23
- /* harmony export */ "augmentWebsocket": () => (/* binding */ augmentWebsocket)
24
- /* harmony export */ });
25
- /**
26
- * @internal
27
- */
28
- function augmentWebsocket(webSocket, debug) {
29
- webSocket.terminate = function () {
30
- const noOp = () => { };
31
- // set all callbacks to no op
32
- this.onerror = noOp;
33
- this.onmessage = noOp;
34
- this.onopen = noOp;
35
- const ts = new Date();
36
- const id = Math.random().toString().substring(2, 8); // A simulated id
37
- const origOnClose = this.onclose;
38
- // Track delay in actual closure of the socket
39
- this.onclose = closeEvent => {
40
- const delay = new Date().getTime() - ts.getTime();
41
- debug(`Discarded socket (#${id}) closed after ${delay}ms, with code/reason: ${closeEvent.code}/${closeEvent.reason}`);
42
- };
43
- this.close();
44
- origOnClose === null || origOnClose === void 0 ? void 0 : origOnClose.call(webSocket, {
45
- code: 4001,
46
- reason: `Quick discarding socket (#${id}) without waiting for the shutdown sequence.`,
47
- wasClean: false,
48
- });
7
+ /**
8
+ * Some byte values, used as per STOMP specifications.
9
+ *
10
+ * Part of `@stomp/stompjs`.
11
+ *
12
+ * @internal
13
+ */
14
+ const BYTE = {
15
+ // LINEFEED byte (octet 10)
16
+ LF: '\x0A',
17
+ // NULL byte (octet 0)
18
+ NULL: '\x00',
49
19
  };
50
- }
51
-
52
-
53
- /***/ }),
54
-
55
- /***/ "./src/byte.ts":
56
- /*!*********************!*\
57
- !*** ./src/byte.ts ***!
58
- \*********************/
59
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
60
-
61
- __webpack_require__.r(__webpack_exports__);
62
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
63
- /* harmony export */ "BYTE": () => (/* binding */ BYTE)
64
- /* harmony export */ });
65
- /**
66
- * Some byte values, used as per STOMP specifications.
67
- *
68
- * Part of `@stomp/stompjs`.
69
- *
70
- * @internal
71
- */
72
- const BYTE = {
73
- // LINEFEED byte (octet 10)
74
- LF: '\x0A',
75
- // NULL byte (octet 0)
76
- NULL: '\x00',
77
- };
78
-
79
-
80
- /***/ }),
81
-
82
- /***/ "./src/client.ts":
83
- /*!***********************!*\
84
- !*** ./src/client.ts ***!
85
- \***********************/
86
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
87
-
88
- __webpack_require__.r(__webpack_exports__);
89
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
90
- /* harmony export */ "Client": () => (/* binding */ Client)
91
- /* harmony export */ });
92
- /* harmony import */ var _stomp_handler__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./stomp-handler */ "./src/stomp-handler.ts");
93
- /* harmony import */ var _types__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./types */ "./src/types.ts");
94
- /* harmony import */ var _versions__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./versions */ "./src/versions.ts");
95
- var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {
96
- function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
97
- return new (P || (P = Promise))(function (resolve, reject) {
98
- function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
99
- function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
100
- function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
101
- step((generator = generator.apply(thisArg, _arguments || [])).next());
102
- });
103
- };
104
-
105
20
 
106
-
107
- /**
108
- * STOMP Client Class.
109
- *
110
- * Part of `@stomp/stompjs`.
111
- */
112
- class Client {
113
21
  /**
114
- * Create an instance.
22
+ * Frame class represents a STOMP frame.
23
+ *
24
+ * @internal
115
25
  */
116
- constructor(conf = {}) {
26
+ class FrameImpl {
117
27
  /**
118
- * STOMP versions to attempt during STOMP handshake. By default versions `1.0`, `1.1`, and `1.2` are attempted.
28
+ * Frame constructor. `command`, `headers` and `body` are available as properties.
119
29
  *
120
- * Example:
121
- * ```javascript
122
- * // Try only versions 1.0 and 1.1
123
- * client.stompVersions = new Versions(['1.0', '1.1'])
124
- * ```
30
+ * @internal
125
31
  */
126
- this.stompVersions = _versions__WEBPACK_IMPORTED_MODULE_0__.Versions["default"];
32
+ constructor(params) {
33
+ const { command, headers, body, binaryBody, escapeHeaderValues, skipContentLengthHeader, } = params;
34
+ this.command = command;
35
+ this.headers = Object.assign({}, headers || {});
36
+ if (binaryBody) {
37
+ this._binaryBody = binaryBody;
38
+ this.isBinaryBody = true;
39
+ }
40
+ else {
41
+ this._body = body || '';
42
+ this.isBinaryBody = false;
43
+ }
44
+ this.escapeHeaderValues = escapeHeaderValues || false;
45
+ this.skipContentLengthHeader = skipContentLengthHeader || false;
46
+ }
127
47
  /**
128
- * Will retry if Stomp connection is not established in specified milliseconds.
129
- * Default 0, which implies wait for ever.
48
+ * body of the frame
130
49
  */
131
- this.connectionTimeout = 0;
50
+ get body() {
51
+ if (!this._body && this.isBinaryBody) {
52
+ this._body = new TextDecoder().decode(this._binaryBody);
53
+ }
54
+ return this._body || '';
55
+ }
132
56
  /**
133
- * automatically reconnect with delay in milliseconds, set to 0 to disable.
57
+ * body as Uint8Array
134
58
  */
135
- this.reconnectDelay = 5000;
59
+ get binaryBody() {
60
+ if (!this._binaryBody && !this.isBinaryBody) {
61
+ this._binaryBody = new TextEncoder().encode(this._body);
62
+ }
63
+ // At this stage it will definitely have a valid value
64
+ return this._binaryBody;
65
+ }
136
66
  /**
137
- * Incoming heartbeat interval in milliseconds. Set to 0 to disable.
67
+ * deserialize a STOMP Frame from raw data.
68
+ *
69
+ * @internal
138
70
  */
139
- this.heartbeatIncoming = 10000;
71
+ static fromRawFrame(rawFrame, escapeHeaderValues) {
72
+ const headers = {};
73
+ const trim = (str) => str.replace(/^\s+|\s+$/g, '');
74
+ // In case of repeated headers, as per standards, first value need to be used
75
+ for (const header of rawFrame.headers.reverse()) {
76
+ header.indexOf(':');
77
+ const key = trim(header[0]);
78
+ let value = trim(header[1]);
79
+ if (escapeHeaderValues &&
80
+ rawFrame.command !== 'CONNECT' &&
81
+ rawFrame.command !== 'CONNECTED') {
82
+ value = FrameImpl.hdrValueUnEscape(value);
83
+ }
84
+ headers[key] = value;
85
+ }
86
+ return new FrameImpl({
87
+ command: rawFrame.command,
88
+ headers,
89
+ binaryBody: rawFrame.binaryBody,
90
+ escapeHeaderValues,
91
+ });
92
+ }
140
93
  /**
141
- * Outgoing heartbeat interval in milliseconds. Set to 0 to disable.
94
+ * @internal
142
95
  */
143
- this.heartbeatOutgoing = 10000;
96
+ toString() {
97
+ return this.serializeCmdAndHeaders();
98
+ }
144
99
  /**
145
- * This switches on a non standard behavior while sending WebSocket packets.
146
- * It splits larger (text) packets into chunks of [maxWebSocketChunkSize]{@link Client#maxWebSocketChunkSize}.
147
- * Only Java Spring brokers seems to use this mode.
100
+ * serialize this Frame in a format suitable to be passed to WebSocket.
101
+ * If the body is string the output will be string.
102
+ * If the body is binary (i.e. of type Unit8Array) it will be serialized to ArrayBuffer.
148
103
  *
149
- * WebSockets, by itself, split large (text) packets,
150
- * so it is not needed with a truly compliant STOMP/WebSocket broker.
151
- * Actually setting it for such broker will cause large messages to fail.
152
- *
153
- * `false` by default.
154
- *
155
- * Binary frames are never split.
104
+ * @internal
156
105
  */
157
- this.splitLargeFrames = false;
106
+ serialize() {
107
+ const cmdAndHeaders = this.serializeCmdAndHeaders();
108
+ if (this.isBinaryBody) {
109
+ return FrameImpl.toUnit8Array(cmdAndHeaders, this._binaryBody).buffer;
110
+ }
111
+ else {
112
+ return cmdAndHeaders + this._body + BYTE.NULL;
113
+ }
114
+ }
115
+ serializeCmdAndHeaders() {
116
+ const lines = [this.command];
117
+ if (this.skipContentLengthHeader) {
118
+ delete this.headers['content-length'];
119
+ }
120
+ for (const name of Object.keys(this.headers || {})) {
121
+ const value = this.headers[name];
122
+ if (this.escapeHeaderValues &&
123
+ this.command !== 'CONNECT' &&
124
+ this.command !== 'CONNECTED') {
125
+ lines.push(`${name}:${FrameImpl.hdrValueEscape(`${value}`)}`);
126
+ }
127
+ else {
128
+ lines.push(`${name}:${value}`);
129
+ }
130
+ }
131
+ if (this.isBinaryBody ||
132
+ (!this.isBodyEmpty() && !this.skipContentLengthHeader)) {
133
+ lines.push(`content-length:${this.bodyLength()}`);
134
+ }
135
+ return lines.join(BYTE.LF) + BYTE.LF + BYTE.LF;
136
+ }
137
+ isBodyEmpty() {
138
+ return this.bodyLength() === 0;
139
+ }
140
+ bodyLength() {
141
+ const binaryBody = this.binaryBody;
142
+ return binaryBody ? binaryBody.length : 0;
143
+ }
158
144
  /**
159
- * See [splitLargeFrames]{@link Client#splitLargeFrames}.
160
- * This has no effect if [splitLargeFrames]{@link Client#splitLargeFrames} is `false`.
145
+ * Compute the size of a UTF-8 string by counting its number of bytes
146
+ * (and not the number of characters composing the string)
161
147
  */
162
- this.maxWebSocketChunkSize = 8 * 1024;
148
+ static sizeOfUTF8(s) {
149
+ return s ? new TextEncoder().encode(s).length : 0;
150
+ }
151
+ static toUnit8Array(cmdAndHeaders, binaryBody) {
152
+ const uint8CmdAndHeaders = new TextEncoder().encode(cmdAndHeaders);
153
+ const nullTerminator = new Uint8Array([0]);
154
+ const uint8Frame = new Uint8Array(uint8CmdAndHeaders.length + binaryBody.length + nullTerminator.length);
155
+ uint8Frame.set(uint8CmdAndHeaders);
156
+ uint8Frame.set(binaryBody, uint8CmdAndHeaders.length);
157
+ uint8Frame.set(nullTerminator, uint8CmdAndHeaders.length + binaryBody.length);
158
+ return uint8Frame;
159
+ }
163
160
  /**
164
- * Usually the
165
- * [type of WebSocket frame]{@link https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/send#Parameters}
166
- * is automatically decided by type of the payload.
167
- * Default is `false`, which should work with all compliant brokers.
161
+ * Serialize a STOMP frame as per STOMP standards, suitable to be sent to the STOMP broker.
168
162
  *
169
- * Set this flag to force binary frames.
163
+ * @internal
170
164
  */
171
- this.forceBinaryWSFrames = false;
172
- /**
173
- * A bug in ReactNative chops a string on occurrence of a NULL.
174
- * See issue [https://github.com/stomp-js/stompjs/issues/89]{@link https://github.com/stomp-js/stompjs/issues/89}.
175
- * This makes incoming WebSocket messages invalid STOMP packets.
176
- * Setting this flag attempts to reverse the damage by appending a NULL.
177
- * If the broker splits a large message into multiple WebSocket messages,
178
- * this flag will cause data loss and abnormal termination of connection.
179
- *
180
- * This is not an ideal solution, but a stop gap until the underlying issue is fixed at ReactNative library.
181
- */
182
- this.appendMissingNULLonIncoming = false;
165
+ static marshall(params) {
166
+ const frame = new FrameImpl(params);
167
+ return frame.serialize();
168
+ }
183
169
  /**
184
- * Browsers do not immediately close WebSockets when `.close` is issued.
185
- * This may cause reconnection to take a longer on certain type of failures.
186
- * In case of incoming heartbeat failure, this experimental flag instructs the library
187
- * to discard the socket immediately (even before it is actually closed).
170
+ * Escape header values
188
171
  */
189
- this.discardWebsocketOnCommFailure = false;
172
+ static hdrValueEscape(str) {
173
+ return str
174
+ .replace(/\\/g, '\\\\')
175
+ .replace(/\r/g, '\\r')
176
+ .replace(/\n/g, '\\n')
177
+ .replace(/:/g, '\\c');
178
+ }
190
179
  /**
191
- * Activation state.
192
- *
193
- * It will usually be ACTIVE or INACTIVE.
194
- * When deactivating it may go from ACTIVE to INACTIVE without entering DEACTIVATING.
180
+ * UnEscape header values
195
181
  */
196
- this.state = _types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.INACTIVE;
197
- // Dummy callbacks
198
- const noOp = () => { };
199
- this.debug = noOp;
200
- this.beforeConnect = noOp;
201
- this.onConnect = noOp;
202
- this.onDisconnect = noOp;
203
- this.onUnhandledMessage = noOp;
204
- this.onUnhandledReceipt = noOp;
205
- this.onUnhandledFrame = noOp;
206
- this.onStompError = noOp;
207
- this.onWebSocketClose = noOp;
208
- this.onWebSocketError = noOp;
209
- this.logRawCommunication = false;
210
- this.onChangeState = noOp;
211
- // These parameters would typically get proper values before connect is called
212
- this.connectHeaders = {};
213
- this._disconnectHeaders = {};
214
- // Apply configuration
215
- this.configure(conf);
216
- }
217
- /**
218
- * Underlying WebSocket instance, READONLY.
219
- */
220
- get webSocket() {
221
- var _a;
222
- return (_a = this._stompHandler) === null || _a === void 0 ? void 0 : _a._webSocket;
223
- }
224
- /**
225
- * Disconnection headers.
226
- */
227
- get disconnectHeaders() {
228
- return this._disconnectHeaders;
229
- }
230
- set disconnectHeaders(value) {
231
- this._disconnectHeaders = value;
232
- if (this._stompHandler) {
233
- this._stompHandler.disconnectHeaders = this._disconnectHeaders;
182
+ static hdrValueUnEscape(str) {
183
+ return str
184
+ .replace(/\\r/g, '\r')
185
+ .replace(/\\n/g, '\n')
186
+ .replace(/\\c/g, ':')
187
+ .replace(/\\\\/g, '\\');
234
188
  }
235
189
  }
190
+
236
191
  /**
237
- * `true` if there is a active connection with STOMP Broker
192
+ * @internal
238
193
  */
239
- get connected() {
240
- return !!this._stompHandler && this._stompHandler.connected;
241
- }
194
+ const NULL = 0;
242
195
  /**
243
- * version of STOMP protocol negotiated with the server, READONLY
196
+ * @internal
244
197
  */
245
- get connectedVersion() {
246
- return this._stompHandler ? this._stompHandler.connectedVersion : undefined;
247
- }
198
+ const LF = 10;
248
199
  /**
249
- * if the client is active (connected or going to reconnect)
200
+ * @internal
250
201
  */
251
- get active() {
252
- return this.state === _types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.ACTIVE;
253
- }
254
- _changeState(state) {
255
- this.state = state;
256
- this.onChangeState(state);
257
- }
202
+ const CR = 13;
258
203
  /**
259
- * Update configuration.
204
+ * @internal
260
205
  */
261
- configure(conf) {
262
- // bulk assign all properties to this
263
- Object.assign(this, conf);
264
- }
206
+ const COLON = 58;
265
207
  /**
266
- * Initiate the connection with the broker.
267
- * If the connection breaks, as per [Client#reconnectDelay]{@link Client#reconnectDelay},
268
- * it will keep trying to reconnect.
208
+ * This is an evented, rec descent parser.
209
+ * A stream of Octets can be passed and whenever it recognizes
210
+ * a complete Frame or an incoming ping it will invoke the registered callbacks.
211
+ *
212
+ * All incoming Octets are fed into _onByte function.
213
+ * Depending on current state the _onByte function keeps changing.
214
+ * Depending on the state it keeps accumulating into _token and _results.
215
+ * State is indicated by current value of _onByte, all states are named as _collect.
216
+ *
217
+ * STOMP standards https://stomp.github.io/stomp-specification-1.2.html
218
+ * imply that all lengths are considered in bytes (instead of string lengths).
219
+ * So, before actual parsing, if the incoming data is String it is converted to Octets.
220
+ * This allows faithful implementation of the protocol and allows NULL Octets to be present in the body.
221
+ *
222
+ * There is no peek function on the incoming data.
223
+ * When a state change occurs based on an Octet without consuming the Octet,
224
+ * the Octet, after state change, is fed again (_reinjectByte).
225
+ * This became possible as the state change can be determined by inspecting just one Octet.
226
+ *
227
+ * There are two modes to collect the body, if content-length header is there then it by counting Octets
228
+ * otherwise it is determined by NULL terminator.
229
+ *
230
+ * Following the standards, the command and headers are converted to Strings
231
+ * and the body is returned as Octets.
232
+ * Headers are returned as an array and not as Hash - to allow multiple occurrence of an header.
233
+ *
234
+ * This parser does not use Regular Expressions as that can only operate on Strings.
269
235
  *
270
- * Call [Client#deactivate]{@link Client#deactivate} to disconnect and stop reconnection attempts.
236
+ * It handles if multiple STOMP frames are given as one chunk, a frame is split into multiple chunks, or
237
+ * any combination there of. The parser remembers its state (any partial frame) and continues when a new chunk
238
+ * is pushed.
239
+ *
240
+ * Typically the higher level function will convert headers to Hash, handle unescaping of header values
241
+ * (which is protocol version specific), and convert body to text.
242
+ *
243
+ * Check the parser.spec.js to understand cases that this parser is supposed to handle.
244
+ *
245
+ * Part of `@stomp/stompjs`.
246
+ *
247
+ * @internal
271
248
  */
272
- activate() {
273
- if (this.state === _types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.DEACTIVATING) {
274
- this.debug('Still DEACTIVATING, please await call to deactivate before trying to re-activate');
275
- throw new Error('Still DEACTIVATING, can not activate now');
276
- }
277
- if (this.active) {
278
- this.debug('Already ACTIVE, ignoring request to activate');
279
- return;
249
+ class Parser {
250
+ constructor(onFrame, onIncomingPing) {
251
+ this.onFrame = onFrame;
252
+ this.onIncomingPing = onIncomingPing;
253
+ this._encoder = new TextEncoder();
254
+ this._decoder = new TextDecoder();
255
+ this._token = [];
256
+ this._initState();
257
+ }
258
+ parseChunk(segment, appendMissingNULLonIncoming = false) {
259
+ let chunk;
260
+ if (segment instanceof ArrayBuffer) {
261
+ chunk = new Uint8Array(segment);
262
+ }
263
+ else {
264
+ chunk = this._encoder.encode(segment);
265
+ }
266
+ // See https://github.com/stomp-js/stompjs/issues/89
267
+ // Remove when underlying issue is fixed.
268
+ //
269
+ // Send a NULL byte, if the last byte of a Text frame was not NULL.F
270
+ if (appendMissingNULLonIncoming && chunk[chunk.length - 1] !== 0) {
271
+ const chunkWithNull = new Uint8Array(chunk.length + 1);
272
+ chunkWithNull.set(chunk, 0);
273
+ chunkWithNull[chunk.length] = 0;
274
+ chunk = chunkWithNull;
275
+ }
276
+ // tslint:disable-next-line:prefer-for-of
277
+ for (let i = 0; i < chunk.length; i++) {
278
+ const byte = chunk[i];
279
+ this._onByte(byte);
280
+ }
280
281
  }
281
- this._changeState(_types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.ACTIVE);
282
- this._connect();
283
- }
284
- _connect() {
285
- return __awaiter(this, void 0, void 0, function* () {
286
- if (this.connected) {
287
- this.debug('STOMP: already connected, nothing to do');
282
+ // The following implements a simple Rec Descent Parser.
283
+ // The grammar is simple and just one byte tells what should be the next state
284
+ _collectFrame(byte) {
285
+ if (byte === NULL) {
286
+ // Ignore
288
287
  return;
289
288
  }
290
- yield this.beforeConnect();
291
- if (!this.active) {
292
- this.debug('Client has been marked inactive, will not attempt to connect');
289
+ if (byte === CR) {
290
+ // Ignore CR
293
291
  return;
294
292
  }
295
- // setup connection watcher
296
- if (this.connectionTimeout > 0) {
297
- // clear first
298
- if (this._connectionWatcher) {
299
- clearTimeout(this._connectionWatcher);
300
- }
301
- this._connectionWatcher = setTimeout(() => {
302
- if (this.connected) {
303
- return;
304
- }
305
- // Connection not established, close the underlying socket
306
- // a reconnection will be attempted
307
- this.debug(`Connection not established in ${this.connectionTimeout}ms, closing socket`);
308
- this.forceDisconnect();
309
- }, this.connectionTimeout);
293
+ if (byte === LF) {
294
+ // Incoming Ping
295
+ this.onIncomingPing();
296
+ return;
310
297
  }
311
- this.debug('Opening Web Socket...');
312
- // Get the actual WebSocket (or a similar object)
313
- const webSocket = this._createWebSocket();
314
- this._stompHandler = new _stomp_handler__WEBPACK_IMPORTED_MODULE_2__.StompHandler(this, webSocket, {
315
- debug: this.debug,
316
- stompVersions: this.stompVersions,
317
- connectHeaders: this.connectHeaders,
318
- disconnectHeaders: this._disconnectHeaders,
319
- heartbeatIncoming: this.heartbeatIncoming,
320
- heartbeatOutgoing: this.heartbeatOutgoing,
321
- splitLargeFrames: this.splitLargeFrames,
322
- maxWebSocketChunkSize: this.maxWebSocketChunkSize,
323
- forceBinaryWSFrames: this.forceBinaryWSFrames,
324
- logRawCommunication: this.logRawCommunication,
325
- appendMissingNULLonIncoming: this.appendMissingNULLonIncoming,
326
- discardWebsocketOnCommFailure: this.discardWebsocketOnCommFailure,
327
- onConnect: frame => {
328
- // Successfully connected, stop the connection watcher
329
- if (this._connectionWatcher) {
330
- clearTimeout(this._connectionWatcher);
331
- this._connectionWatcher = undefined;
332
- }
333
- if (!this.active) {
334
- this.debug('STOMP got connected while deactivate was issued, will disconnect now');
335
- this._disposeStompHandler();
336
- return;
337
- }
338
- this.onConnect(frame);
339
- },
340
- onDisconnect: frame => {
341
- this.onDisconnect(frame);
342
- },
343
- onStompError: frame => {
344
- this.onStompError(frame);
345
- },
346
- onWebSocketClose: evt => {
347
- this._stompHandler = undefined; // a new one will be created in case of a reconnect
348
- if (this.state === _types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.DEACTIVATING) {
349
- // Mark deactivation complete
350
- this._changeState(_types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.INACTIVE);
351
- }
352
- // The callback is called before attempting to reconnect, this would allow the client
353
- // to be `deactivated` in the callback.
354
- this.onWebSocketClose(evt);
355
- if (this.active) {
356
- this._schedule_reconnect();
357
- }
358
- },
359
- onWebSocketError: evt => {
360
- this.onWebSocketError(evt);
361
- },
362
- onUnhandledMessage: message => {
363
- this.onUnhandledMessage(message);
364
- },
365
- onUnhandledReceipt: frame => {
366
- this.onUnhandledReceipt(frame);
367
- },
368
- onUnhandledFrame: frame => {
369
- this.onUnhandledFrame(frame);
370
- },
371
- });
372
- this._stompHandler.start();
373
- });
374
- }
375
- _createWebSocket() {
376
- let webSocket;
377
- if (this.webSocketFactory) {
378
- webSocket = this.webSocketFactory();
298
+ this._onByte = this._collectCommand;
299
+ this._reinjectByte(byte);
379
300
  }
380
- else if (this.brokerURL) {
381
- webSocket = new WebSocket(this.brokerURL, this.stompVersions.protocolVersions());
301
+ _collectCommand(byte) {
302
+ if (byte === CR) {
303
+ // Ignore CR
304
+ return;
305
+ }
306
+ if (byte === LF) {
307
+ this._results.command = this._consumeTokenAsUTF8();
308
+ this._onByte = this._collectHeaders;
309
+ return;
310
+ }
311
+ this._consumeByte(byte);
382
312
  }
383
- else {
384
- throw new Error('Either brokerURL or webSocketFactory must be provided');
313
+ _collectHeaders(byte) {
314
+ if (byte === CR) {
315
+ // Ignore CR
316
+ return;
317
+ }
318
+ if (byte === LF) {
319
+ this._setupCollectBody();
320
+ return;
321
+ }
322
+ this._onByte = this._collectHeaderKey;
323
+ this._reinjectByte(byte);
385
324
  }
386
- webSocket.binaryType = 'arraybuffer';
387
- return webSocket;
388
- }
389
- _schedule_reconnect() {
390
- if (this.reconnectDelay > 0) {
391
- this.debug(`STOMP: scheduling reconnection in ${this.reconnectDelay}ms`);
392
- this._reconnector = setTimeout(() => {
393
- this._connect();
394
- }, this.reconnectDelay);
325
+ _reinjectByte(byte) {
326
+ this._onByte(byte);
395
327
  }
396
- }
397
- /**
398
- * Disconnect if connected and stop auto reconnect loop.
399
- * Appropriate callbacks will be invoked if there is an underlying STOMP connection.
400
- *
401
- * This call is async. It will resolve immediately if there is no underlying active websocket,
402
- * otherwise, it will resolve after the underlying websocket is properly disposed of.
403
- *
404
- * It is not an error to invoke this method more than once.
405
- * Each of those would resolve on completion of deactivation.
406
- *
407
- * To reactivate, you can call [Client#activate]{@link Client#activate}.
408
- *
409
- * Experimental: pass `force: true` to immediately discard the underlying connection.
410
- * This mode will skip both the STOMP and the Websocket shutdown sequences.
411
- * In some cases, browsers take a long time in the Websocket shutdown if the underlying connection had gone stale.
412
- * Using this mode can speed up.
413
- * When this mode is used, the actual Websocket may linger for a while
414
- * and the broker may not realize that the connection is no longer in use.
415
- *
416
- * It is possible to invoke this method initially without the `force` option
417
- * and subsequently, say after a wait, with the `force` option.
418
- */
419
- deactivate(options = {}) {
420
- var _a;
421
- return __awaiter(this, void 0, void 0, function* () {
422
- const force = options.force || false;
423
- const needToDispose = this.active;
424
- let retPromise;
425
- if (this.state === _types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.INACTIVE) {
426
- this.debug(`Already INACTIVE, nothing more to do`);
427
- return Promise.resolve();
328
+ _collectHeaderKey(byte) {
329
+ if (byte === COLON) {
330
+ this._headerKey = this._consumeTokenAsUTF8();
331
+ this._onByte = this._collectHeaderValue;
332
+ return;
428
333
  }
429
- this._changeState(_types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.DEACTIVATING);
430
- // Clear if a reconnection was scheduled
431
- if (this._reconnector) {
432
- clearTimeout(this._reconnector);
433
- this._reconnector = undefined;
334
+ this._consumeByte(byte);
335
+ }
336
+ _collectHeaderValue(byte) {
337
+ if (byte === CR) {
338
+ // Ignore CR
339
+ return;
434
340
  }
435
- if (this._stompHandler &&
436
- // @ts-ignore - if there is a _stompHandler, there is the webSocket
437
- this.webSocket.readyState !== _types__WEBPACK_IMPORTED_MODULE_1__.StompSocketState.CLOSED) {
438
- const origOnWebSocketClose = this._stompHandler.onWebSocketClose;
439
- // we need to wait for the underlying websocket to close
440
- retPromise = new Promise((resolve, reject) => {
441
- // @ts-ignore - there is a _stompHandler
442
- this._stompHandler.onWebSocketClose = evt => {
443
- origOnWebSocketClose(evt);
444
- resolve();
445
- };
446
- });
341
+ if (byte === LF) {
342
+ this._results.headers.push([
343
+ this._headerKey,
344
+ this._consumeTokenAsUTF8(),
345
+ ]);
346
+ this._headerKey = undefined;
347
+ this._onByte = this._collectHeaders;
348
+ return;
349
+ }
350
+ this._consumeByte(byte);
351
+ }
352
+ _setupCollectBody() {
353
+ const contentLengthHeader = this._results.headers.filter((header) => {
354
+ return header[0] === 'content-length';
355
+ })[0];
356
+ if (contentLengthHeader) {
357
+ this._bodyBytesRemaining = parseInt(contentLengthHeader[1], 10);
358
+ this._onByte = this._collectBodyFixedSize;
447
359
  }
448
360
  else {
449
- // indicate that auto reconnect loop should terminate
450
- this._changeState(_types__WEBPACK_IMPORTED_MODULE_1__.ActivationState.INACTIVE);
451
- return Promise.resolve();
361
+ this._onByte = this._collectBodyNullTerminated;
452
362
  }
453
- if (force) {
454
- (_a = this._stompHandler) === null || _a === void 0 ? void 0 : _a.discardWebsocket();
363
+ }
364
+ _collectBodyNullTerminated(byte) {
365
+ if (byte === NULL) {
366
+ this._retrievedBody();
367
+ return;
455
368
  }
456
- else if (needToDispose) {
457
- this._disposeStompHandler();
369
+ this._consumeByte(byte);
370
+ }
371
+ _collectBodyFixedSize(byte) {
372
+ // It is post decrement, so that we discard the trailing NULL octet
373
+ if (this._bodyBytesRemaining-- === 0) {
374
+ this._retrievedBody();
375
+ return;
458
376
  }
459
- return retPromise;
460
- });
461
- }
462
- /**
463
- * Force disconnect if there is an active connection by directly closing the underlying WebSocket.
464
- * This is different than a normal disconnect where a DISCONNECT sequence is carried out with the broker.
465
- * After forcing disconnect, automatic reconnect will be attempted.
466
- * To stop further reconnects call [Client#deactivate]{@link Client#deactivate} as well.
467
- */
468
- forceDisconnect() {
469
- if (this._stompHandler) {
470
- this._stompHandler.forceDisconnect();
377
+ this._consumeByte(byte);
471
378
  }
472
- }
473
- _disposeStompHandler() {
474
- // Dispose STOMP Handler
475
- if (this._stompHandler) {
476
- this._stompHandler.dispose();
379
+ _retrievedBody() {
380
+ this._results.binaryBody = this._consumeTokenAsRaw();
381
+ try {
382
+ this.onFrame(this._results);
383
+ }
384
+ catch (e) {
385
+ console.log(`Ignoring an exception thrown by a frame handler. Original exception: `, e);
386
+ }
387
+ this._initState();
388
+ }
389
+ // Rec Descent Parser helpers
390
+ _consumeByte(byte) {
391
+ this._token.push(byte);
392
+ }
393
+ _consumeTokenAsUTF8() {
394
+ return this._decoder.decode(this._consumeTokenAsRaw());
395
+ }
396
+ _consumeTokenAsRaw() {
397
+ const rawResult = new Uint8Array(this._token);
398
+ this._token = [];
399
+ return rawResult;
400
+ }
401
+ _initState() {
402
+ this._results = {
403
+ command: undefined,
404
+ headers: [],
405
+ binaryBody: undefined,
406
+ };
407
+ this._token = [];
408
+ this._headerKey = undefined;
409
+ this._onByte = this._collectFrame;
477
410
  }
478
411
  }
412
+
479
413
  /**
480
- * Send a message to a named destination. Refer to your STOMP broker documentation for types
481
- * and naming of destinations.
482
- *
483
- * STOMP protocol specifies and suggests some headers and also allows broker specific headers.
484
- *
485
- * `body` must be String.
486
- * You will need to covert the payload to string in case it is not string (e.g. JSON).
487
- *
488
- * To send a binary message body use binaryBody parameter. It should be a
489
- * [Uint8Array](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Uint8Array).
490
- * Sometimes brokers may not support binary frames out of the box.
491
- * Please check your broker documentation.
492
- *
493
- * `content-length` header is automatically added to the STOMP Frame sent to the broker.
494
- * Set `skipContentLengthHeader` to indicate that `content-length` header should not be added.
495
- * For binary messages `content-length` header is always added.
496
- *
497
- * Caution: The broker will, most likely, report an error and disconnect if message body has NULL octet(s)
498
- * and `content-length` header is missing.
499
- *
500
- * ```javascript
501
- * client.publish({destination: "/queue/test", headers: {priority: 9}, body: "Hello, STOMP"});
502
- *
503
- * // Only destination is mandatory parameter
504
- * client.publish({destination: "/queue/test", body: "Hello, STOMP"});
505
- *
506
- * // Skip content-length header in the frame to the broker
507
- * client.publish({"/queue/test", body: "Hello, STOMP", skipContentLengthHeader: true});
508
- *
509
- * var binaryData = generateBinaryData(); // This need to be of type Uint8Array
510
- * // setting content-type header is not mandatory, however a good practice
511
- * client.publish({destination: '/topic/special', binaryBody: binaryData,
512
- * headers: {'content-type': 'application/octet-stream'}});
513
- * ```
414
+ * Possible states for the IStompSocket
514
415
  */
515
- publish(params) {
516
- this._checkConnection();
517
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
518
- this._stompHandler.publish(params);
519
- }
520
- _checkConnection() {
521
- if (!this.connected) {
522
- throw new TypeError('There is no underlying STOMP connection');
523
- }
524
- }
416
+ exports.StompSocketState = void 0;
417
+ (function (StompSocketState) {
418
+ StompSocketState[StompSocketState["CONNECTING"] = 0] = "CONNECTING";
419
+ StompSocketState[StompSocketState["OPEN"] = 1] = "OPEN";
420
+ StompSocketState[StompSocketState["CLOSING"] = 2] = "CLOSING";
421
+ StompSocketState[StompSocketState["CLOSED"] = 3] = "CLOSED";
422
+ })(exports.StompSocketState = exports.StompSocketState || (exports.StompSocketState = {}));
525
423
  /**
526
- * STOMP brokers may carry out operation asynchronously and allow requesting for acknowledgement.
527
- * To request an acknowledgement, a `receipt` header needs to be sent with the actual request.
528
- * The value (say receipt-id) for this header needs to be unique for each use. Typically a sequence, a UUID, a
529
- * random number or a combination may be used.
530
- *
531
- * A complaint broker will send a RECEIPT frame when an operation has actually been completed.
532
- * The operation needs to be matched based in the value of the receipt-id.
533
- *
534
- * This method allow watching for a receipt and invoke the callback
535
- * when corresponding receipt has been received.
536
- *
537
- * The actual {@link FrameImpl} will be passed as parameter to the callback.
538
- *
539
- * Example:
540
- * ```javascript
541
- * // Subscribing with acknowledgement
542
- * let receiptId = randomText();
424
+ * Possible activation state
425
+ */
426
+ exports.ActivationState = void 0;
427
+ (function (ActivationState) {
428
+ ActivationState[ActivationState["ACTIVE"] = 0] = "ACTIVE";
429
+ ActivationState[ActivationState["DEACTIVATING"] = 1] = "DEACTIVATING";
430
+ ActivationState[ActivationState["INACTIVE"] = 2] = "INACTIVE";
431
+ })(exports.ActivationState = exports.ActivationState || (exports.ActivationState = {}));
432
+
433
+ /**
434
+ * Supported STOMP versions
543
435
  *
544
- * client.watchForReceipt(receiptId, function() {
545
- * // Will be called after server acknowledges
546
- * });
547
- *
548
- * client.subscribe(TEST.destination, onMessage, {receipt: receiptId});
549
- *
550
- *
551
- * // Publishing with acknowledgement
552
- * receiptId = randomText();
553
- *
554
- * client.watchForReceipt(receiptId, function() {
555
- * // Will be called after server acknowledges
556
- * });
557
- * client.publish({destination: TEST.destination, headers: {receipt: receiptId}, body: msg});
558
- * ```
559
- */
560
- watchForReceipt(receiptId, callback) {
561
- this._checkConnection();
562
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
563
- this._stompHandler.watchForReceipt(receiptId, callback);
564
- }
565
- /**
566
- * Subscribe to a STOMP Broker location. The callback will be invoked for each received message with
567
- * the {@link IMessage} as argument.
568
- *
569
- * Note: The library will generate an unique ID if there is none provided in the headers.
570
- * To use your own ID, pass it using the headers argument.
571
- *
572
- * ```javascript
573
- * callback = function(message) {
574
- * // called when the client receives a STOMP message from the server
575
- * if (message.body) {
576
- * alert("got message with body " + message.body)
577
- * } else {
578
- * alert("got empty message");
579
- * }
580
- * });
581
- *
582
- * var subscription = client.subscribe("/queue/test", callback);
583
- *
584
- * // Explicit subscription id
585
- * var mySubId = 'my-subscription-id-001';
586
- * var subscription = client.subscribe(destination, callback, { id: mySubId });
587
- * ```
588
- */
589
- subscribe(destination, callback, headers = {}) {
590
- this._checkConnection();
591
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
592
- return this._stompHandler.subscribe(destination, callback, headers);
593
- }
594
- /**
595
- * It is preferable to unsubscribe from a subscription by calling
596
- * `unsubscribe()` directly on {@link StompSubscription} returned by `client.subscribe()`:
597
- *
598
- * ```javascript
599
- * var subscription = client.subscribe(destination, onmessage);
600
- * // ...
601
- * subscription.unsubscribe();
602
- * ```
603
- *
604
- * See: http://stomp.github.com/stomp-specification-1.2.html#UNSUBSCRIBE UNSUBSCRIBE Frame
605
- */
606
- unsubscribe(id, headers = {}) {
607
- this._checkConnection();
608
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
609
- this._stompHandler.unsubscribe(id, headers);
610
- }
611
- /**
612
- * Start a transaction, the returned {@link ITransaction} has methods - [commit]{@link ITransaction#commit}
613
- * and [abort]{@link ITransaction#abort}.
614
- *
615
- * `transactionId` is optional, if not passed the library will generate it internally.
436
+ * Part of `@stomp/stompjs`.
616
437
  */
617
- begin(transactionId) {
618
- this._checkConnection();
619
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
620
- return this._stompHandler.begin(transactionId);
438
+ class Versions {
439
+ /**
440
+ * Takes an array of string of versions, typical elements '1.0', '1.1', or '1.2'
441
+ *
442
+ * You will an instance if this class if you want to override supported versions to be declared during
443
+ * STOMP handshake.
444
+ */
445
+ constructor(versions) {
446
+ this.versions = versions;
447
+ }
448
+ /**
449
+ * Used as part of CONNECT STOMP Frame
450
+ */
451
+ supportedVersions() {
452
+ return this.versions.join(',');
453
+ }
454
+ /**
455
+ * Used while creating a WebSocket
456
+ */
457
+ protocolVersions() {
458
+ return this.versions.map(x => `v${x.replace('.', '')}.stomp`);
459
+ }
621
460
  }
622
461
  /**
623
- * Commit a transaction.
624
- *
625
- * It is preferable to commit a transaction by calling [commit]{@link ITransaction#commit} directly on
626
- * {@link ITransaction} returned by [client.begin]{@link Client#begin}.
627
- *
628
- * ```javascript
629
- * var tx = client.begin(txId);
630
- * //...
631
- * tx.commit();
632
- * ```
462
+ * Indicates protocol version 1.0
633
463
  */
634
- commit(transactionId) {
635
- this._checkConnection();
636
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
637
- this._stompHandler.commit(transactionId);
638
- }
464
+ Versions.V1_0 = '1.0';
639
465
  /**
640
- * Abort a transaction.
641
- * It is preferable to abort a transaction by calling [abort]{@link ITransaction#abort} directly on
642
- * {@link ITransaction} returned by [client.begin]{@link Client#begin}.
643
- *
644
- * ```javascript
645
- * var tx = client.begin(txId);
646
- * //...
647
- * tx.abort();
648
- * ```
466
+ * Indicates protocol version 1.1
649
467
  */
650
- abort(transactionId) {
651
- this._checkConnection();
652
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
653
- this._stompHandler.abort(transactionId);
654
- }
468
+ Versions.V1_1 = '1.1';
655
469
  /**
656
- * ACK a message. It is preferable to acknowledge a message by calling [ack]{@link IMessage#ack} directly
657
- * on the {@link IMessage} handled by a subscription callback:
658
- *
659
- * ```javascript
660
- * var callback = function (message) {
661
- * // process the message
662
- * // acknowledge it
663
- * message.ack();
664
- * };
665
- * client.subscribe(destination, callback, {'ack': 'client'});
666
- * ```
470
+ * Indicates protocol version 1.2
667
471
  */
668
- ack(messageId, subscriptionId, headers = {}) {
669
- this._checkConnection();
670
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
671
- this._stompHandler.ack(messageId, subscriptionId, headers);
672
- }
472
+ Versions.V1_2 = '1.2';
673
473
  /**
674
- * NACK a message. It is preferable to acknowledge a message by calling [nack]{@link IMessage#nack} directly
675
- * on the {@link IMessage} handled by a subscription callback:
676
- *
677
- * ```javascript
678
- * var callback = function (message) {
679
- * // process the message
680
- * // an error occurs, nack it
681
- * message.nack();
682
- * };
683
- * client.subscribe(destination, callback, {'ack': 'client'});
684
- * ```
474
+ * @internal
685
475
  */
686
- nack(messageId, subscriptionId, headers = {}) {
687
- this._checkConnection();
688
- // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
689
- this._stompHandler.nack(messageId, subscriptionId, headers);
690
- }
691
- }
692
-
693
-
694
- /***/ }),
695
-
696
- /***/ "./src/compatibility/compat-client.ts":
697
- /*!********************************************!*\
698
- !*** ./src/compatibility/compat-client.ts ***!
699
- \********************************************/
700
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
476
+ Versions.default = new Versions([
477
+ Versions.V1_2,
478
+ Versions.V1_1,
479
+ Versions.V1_0,
480
+ ]);
701
481
 
702
- __webpack_require__.r(__webpack_exports__);
703
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
704
- /* harmony export */ "CompatClient": () => (/* binding */ CompatClient)
705
- /* harmony export */ });
706
- /* harmony import */ var _client__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../client */ "./src/client.ts");
707
- /* harmony import */ var _heartbeat_info__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./heartbeat-info */ "./src/compatibility/heartbeat-info.ts");
708
-
709
-
710
- /**
711
- * Available for backward compatibility, please shift to using {@link Client}.
712
- *
713
- * **Deprecated**
714
- *
715
- * Part of `@stomp/stompjs`.
716
- *
717
- * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)
718
- */
719
- class CompatClient extends _client__WEBPACK_IMPORTED_MODULE_0__.Client {
720
482
  /**
721
- * Available for backward compatibility, please shift to using {@link Client}
722
- * and [Client#webSocketFactory]{@link Client#webSocketFactory}.
723
- *
724
- * **Deprecated**
725
- *
726
483
  * @internal
727
484
  */
728
- constructor(webSocketFactory) {
729
- super();
730
- /**
731
- * It is no op now. No longer needed. Large packets work out of the box.
732
- */
733
- this.maxWebSocketFrameSize = 16 * 1024;
734
- this._heartbeatInfo = new _heartbeat_info__WEBPACK_IMPORTED_MODULE_1__.HeartbeatInfo(this);
735
- this.reconnect_delay = 0;
736
- this.webSocketFactory = webSocketFactory;
737
- // Default from previous version
738
- this.debug = (...message) => {
739
- console.log(...message);
485
+ function augmentWebsocket(webSocket, debug) {
486
+ webSocket.terminate = function () {
487
+ const noOp = () => { };
488
+ // set all callbacks to no op
489
+ this.onerror = noOp;
490
+ this.onmessage = noOp;
491
+ this.onopen = noOp;
492
+ const ts = new Date();
493
+ const id = Math.random().toString().substring(2, 8); // A simulated id
494
+ const origOnClose = this.onclose;
495
+ // Track delay in actual closure of the socket
496
+ this.onclose = closeEvent => {
497
+ const delay = new Date().getTime() - ts.getTime();
498
+ debug(`Discarded socket (#${id}) closed after ${delay}ms, with code/reason: ${closeEvent.code}/${closeEvent.reason}`);
499
+ };
500
+ this.close();
501
+ origOnClose?.call(webSocket, {
502
+ code: 4001,
503
+ reason: `Quick discarding socket (#${id}) without waiting for the shutdown sequence.`,
504
+ wasClean: false,
505
+ });
740
506
  };
741
507
  }
742
- _parseConnect(...args) {
743
- let closeEventCallback;
744
- let connectCallback;
745
- let errorCallback;
746
- let headers = {};
747
- if (args.length < 2) {
748
- throw new Error('Connect requires at least 2 arguments');
749
- }
750
- if (typeof args[1] === 'function') {
751
- [headers, connectCallback, errorCallback, closeEventCallback] = args;
752
- }
753
- else {
754
- switch (args.length) {
755
- case 6:
756
- [
757
- headers.login,
758
- headers.passcode,
759
- connectCallback,
760
- errorCallback,
761
- closeEventCallback,
762
- headers.host,
763
- ] = args;
764
- break;
765
- default:
766
- [
767
- headers.login,
768
- headers.passcode,
769
- connectCallback,
770
- errorCallback,
771
- closeEventCallback,
772
- ] = args;
773
- }
774
- }
775
- return [headers, connectCallback, errorCallback, closeEventCallback];
776
- }
508
+
777
509
  /**
778
- * Available for backward compatibility, please shift to using [Client#activate]{@link Client#activate}.
779
- *
780
- * **Deprecated**
781
- *
782
- * The `connect` method accepts different number of arguments and types. See the Overloads list. Use the
783
- * version with headers to pass your broker specific options.
510
+ * The STOMP protocol handler
784
511
  *
785
- * overloads:
786
- * - connect(headers, connectCallback)
787
- * - connect(headers, connectCallback, errorCallback)
788
- * - connect(login, passcode, connectCallback)
789
- * - connect(login, passcode, connectCallback, errorCallback)
790
- * - connect(login, passcode, connectCallback, errorCallback, closeEventCallback)
791
- * - connect(login, passcode, connectCallback, errorCallback, closeEventCallback, host)
512
+ * Part of `@stomp/stompjs`.
792
513
  *
793
- * params:
794
- * - headers, see [Client#connectHeaders]{@link Client#connectHeaders}
795
- * - connectCallback, see [Client#onConnect]{@link Client#onConnect}
796
- * - errorCallback, see [Client#onStompError]{@link Client#onStompError}
797
- * - closeEventCallback, see [Client#onWebSocketClose]{@link Client#onWebSocketClose}
798
- * - login [String], see [Client#connectHeaders](../classes/Client.html#connectHeaders)
799
- * - passcode [String], [Client#connectHeaders](../classes/Client.html#connectHeaders)
800
- * - host [String], see [Client#connectHeaders](../classes/Client.html#connectHeaders)
801
- *
802
- * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)
514
+ * @internal
803
515
  */
804
- connect(...args) {
805
- const out = this._parseConnect(...args);
806
- if (out[0]) {
807
- this.connectHeaders = out[0];
516
+ class StompHandler {
517
+ constructor(_client, _webSocket, config) {
518
+ this._client = _client;
519
+ this._webSocket = _webSocket;
520
+ this._connected = false;
521
+ this._serverFrameHandlers = {
522
+ // [CONNECTED Frame](http://stomp.github.com/stomp-specification-1.2.html#CONNECTED_Frame)
523
+ CONNECTED: frame => {
524
+ this.debug(`connected to server ${frame.headers.server}`);
525
+ this._connected = true;
526
+ this._connectedVersion = frame.headers.version;
527
+ // STOMP version 1.2 needs header values to be escaped
528
+ if (this._connectedVersion === Versions.V1_2) {
529
+ this._escapeHeaderValues = true;
530
+ }
531
+ this._setupHeartbeat(frame.headers);
532
+ this.onConnect(frame);
533
+ },
534
+ // [MESSAGE Frame](http://stomp.github.com/stomp-specification-1.2.html#MESSAGE)
535
+ MESSAGE: frame => {
536
+ // the callback is registered when the client calls
537
+ // `subscribe()`.
538
+ // If there is no registered subscription for the received message,
539
+ // the default `onUnhandledMessage` callback is used that the client can set.
540
+ // This is useful for subscriptions that are automatically created
541
+ // on the browser side (e.g. [RabbitMQ's temporary
542
+ // queues](http://www.rabbitmq.com/stomp.html)).
543
+ const subscription = frame.headers.subscription;
544
+ const onReceive = this._subscriptions[subscription] || this.onUnhandledMessage;
545
+ // bless the frame to be a Message
546
+ const message = frame;
547
+ const client = this;
548
+ const messageId = this._connectedVersion === Versions.V1_2
549
+ ? message.headers.ack
550
+ : message.headers['message-id'];
551
+ // add `ack()` and `nack()` methods directly to the returned frame
552
+ // so that a simple call to `message.ack()` can acknowledge the message.
553
+ message.ack = (headers = {}) => {
554
+ return client.ack(messageId, subscription, headers);
555
+ };
556
+ message.nack = (headers = {}) => {
557
+ return client.nack(messageId, subscription, headers);
558
+ };
559
+ onReceive(message);
560
+ },
561
+ // [RECEIPT Frame](http://stomp.github.com/stomp-specification-1.2.html#RECEIPT)
562
+ RECEIPT: frame => {
563
+ const callback = this._receiptWatchers[frame.headers['receipt-id']];
564
+ if (callback) {
565
+ callback(frame);
566
+ // Server will acknowledge only once, remove the callback
567
+ delete this._receiptWatchers[frame.headers['receipt-id']];
568
+ }
569
+ else {
570
+ this.onUnhandledReceipt(frame);
571
+ }
572
+ },
573
+ // [ERROR Frame](http://stomp.github.com/stomp-specification-1.2.html#ERROR)
574
+ ERROR: frame => {
575
+ this.onStompError(frame);
576
+ },
577
+ };
578
+ // used to index subscribers
579
+ this._counter = 0;
580
+ // subscription callbacks indexed by subscriber's ID
581
+ this._subscriptions = {};
582
+ // receipt-watchers indexed by receipts-ids
583
+ this._receiptWatchers = {};
584
+ this._partialData = '';
585
+ this._escapeHeaderValues = false;
586
+ this._lastServerActivityTS = Date.now();
587
+ this.debug = config.debug;
588
+ this.stompVersions = config.stompVersions;
589
+ this.connectHeaders = config.connectHeaders;
590
+ this.disconnectHeaders = config.disconnectHeaders;
591
+ this.heartbeatIncoming = config.heartbeatIncoming;
592
+ this.heartbeatOutgoing = config.heartbeatOutgoing;
593
+ this.splitLargeFrames = config.splitLargeFrames;
594
+ this.maxWebSocketChunkSize = config.maxWebSocketChunkSize;
595
+ this.forceBinaryWSFrames = config.forceBinaryWSFrames;
596
+ this.logRawCommunication = config.logRawCommunication;
597
+ this.appendMissingNULLonIncoming = config.appendMissingNULLonIncoming;
598
+ this.discardWebsocketOnCommFailure = config.discardWebsocketOnCommFailure;
599
+ this.onConnect = config.onConnect;
600
+ this.onDisconnect = config.onDisconnect;
601
+ this.onStompError = config.onStompError;
602
+ this.onWebSocketClose = config.onWebSocketClose;
603
+ this.onWebSocketError = config.onWebSocketError;
604
+ this.onUnhandledMessage = config.onUnhandledMessage;
605
+ this.onUnhandledReceipt = config.onUnhandledReceipt;
606
+ this.onUnhandledFrame = config.onUnhandledFrame;
607
+ }
608
+ get connectedVersion() {
609
+ return this._connectedVersion;
610
+ }
611
+ get connected() {
612
+ return this._connected;
613
+ }
614
+ start() {
615
+ const parser = new Parser(
616
+ // On Frame
617
+ rawFrame => {
618
+ const frame = FrameImpl.fromRawFrame(rawFrame, this._escapeHeaderValues);
619
+ // if this.logRawCommunication is set, the rawChunk is logged at this._webSocket.onmessage
620
+ if (!this.logRawCommunication) {
621
+ this.debug(`<<< ${frame}`);
622
+ }
623
+ const serverFrameHandler = this._serverFrameHandlers[frame.command] || this.onUnhandledFrame;
624
+ serverFrameHandler(frame);
625
+ },
626
+ // On Incoming Ping
627
+ () => {
628
+ this.debug('<<< PONG');
629
+ });
630
+ this._webSocket.onmessage = (evt) => {
631
+ this.debug('Received data');
632
+ this._lastServerActivityTS = Date.now();
633
+ if (this.logRawCommunication) {
634
+ const rawChunkAsString = evt.data instanceof ArrayBuffer
635
+ ? new TextDecoder().decode(evt.data)
636
+ : evt.data;
637
+ this.debug(`<<< ${rawChunkAsString}`);
638
+ }
639
+ parser.parseChunk(evt.data, this.appendMissingNULLonIncoming);
640
+ };
641
+ this._webSocket.onclose = (closeEvent) => {
642
+ this.debug(`Connection closed to ${this._client.brokerURL}`);
643
+ this._cleanUp();
644
+ this.onWebSocketClose(closeEvent);
645
+ };
646
+ this._webSocket.onerror = (errorEvent) => {
647
+ this.onWebSocketError(errorEvent);
648
+ };
649
+ this._webSocket.onopen = () => {
650
+ // Clone before updating
651
+ const connectHeaders = Object.assign({}, this.connectHeaders);
652
+ this.debug('Web Socket Opened...');
653
+ connectHeaders['accept-version'] = this.stompVersions.supportedVersions();
654
+ connectHeaders['heart-beat'] = [
655
+ this.heartbeatOutgoing,
656
+ this.heartbeatIncoming,
657
+ ].join(',');
658
+ this._transmit({ command: 'CONNECT', headers: connectHeaders });
659
+ };
660
+ }
661
+ _setupHeartbeat(headers) {
662
+ if (headers.version !== Versions.V1_1 &&
663
+ headers.version !== Versions.V1_2) {
664
+ return;
665
+ }
666
+ // It is valid for the server to not send this header
667
+ // https://stomp.github.io/stomp-specification-1.2.html#Heart-beating
668
+ if (!headers['heart-beat']) {
669
+ return;
670
+ }
671
+ // heart-beat header received from the server looks like:
672
+ //
673
+ // heart-beat: sx, sy
674
+ const [serverOutgoing, serverIncoming] = headers['heart-beat']
675
+ .split(',')
676
+ .map((v) => parseInt(v, 10));
677
+ if (this.heartbeatOutgoing !== 0 && serverIncoming !== 0) {
678
+ const ttl = Math.max(this.heartbeatOutgoing, serverIncoming);
679
+ this.debug(`send PING every ${ttl}ms`);
680
+ this._pinger = setInterval(() => {
681
+ if (this._webSocket.readyState === exports.StompSocketState.OPEN) {
682
+ this._webSocket.send(BYTE.LF);
683
+ this.debug('>>> PING');
684
+ }
685
+ }, ttl);
686
+ }
687
+ if (this.heartbeatIncoming !== 0 && serverOutgoing !== 0) {
688
+ const ttl = Math.max(this.heartbeatIncoming, serverOutgoing);
689
+ this.debug(`check PONG every ${ttl}ms`);
690
+ this._ponger = setInterval(() => {
691
+ const delta = Date.now() - this._lastServerActivityTS;
692
+ // We wait twice the TTL to be flexible on window's setInterval calls
693
+ if (delta > ttl * 2) {
694
+ this.debug(`did not receive server activity for the last ${delta}ms`);
695
+ this._closeOrDiscardWebsocket();
696
+ }
697
+ }, ttl);
698
+ }
808
699
  }
809
- if (out[1]) {
810
- this.onConnect = out[1];
700
+ _closeOrDiscardWebsocket() {
701
+ if (this.discardWebsocketOnCommFailure) {
702
+ this.debug('Discarding websocket, the underlying socket may linger for a while');
703
+ this.discardWebsocket();
704
+ }
705
+ else {
706
+ this.debug('Issuing close on the websocket');
707
+ this._closeWebsocket();
708
+ }
811
709
  }
812
- if (out[2]) {
813
- this.onStompError = out[2];
710
+ forceDisconnect() {
711
+ if (this._webSocket) {
712
+ if (this._webSocket.readyState === exports.StompSocketState.CONNECTING ||
713
+ this._webSocket.readyState === exports.StompSocketState.OPEN) {
714
+ this._closeOrDiscardWebsocket();
715
+ }
716
+ }
814
717
  }
815
- if (out[3]) {
816
- this.onWebSocketClose = out[3];
718
+ _closeWebsocket() {
719
+ this._webSocket.onmessage = () => { }; // ignore messages
720
+ this._webSocket.close();
817
721
  }
818
- super.activate();
819
- }
820
- /**
821
- * Available for backward compatibility, please shift to using [Client#deactivate]{@link Client#deactivate}.
822
- *
823
- * **Deprecated**
824
- *
825
- * See:
826
- * [Client#onDisconnect]{@link Client#onDisconnect}, and
827
- * [Client#disconnectHeaders]{@link Client#disconnectHeaders}
828
- *
829
- * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)
830
- */
831
- disconnect(disconnectCallback, headers = {}) {
832
- if (disconnectCallback) {
833
- this.onDisconnect = disconnectCallback;
722
+ discardWebsocket() {
723
+ if (typeof this._webSocket.terminate !== 'function') {
724
+ augmentWebsocket(this._webSocket, (msg) => this.debug(msg));
725
+ }
726
+ // @ts-ignore - this method will be there at this stage
727
+ this._webSocket.terminate();
728
+ }
729
+ _transmit(params) {
730
+ const { command, headers, body, binaryBody, skipContentLengthHeader } = params;
731
+ const frame = new FrameImpl({
732
+ command,
733
+ headers,
734
+ body,
735
+ binaryBody,
736
+ escapeHeaderValues: this._escapeHeaderValues,
737
+ skipContentLengthHeader,
738
+ });
739
+ let rawChunk = frame.serialize();
740
+ if (this.logRawCommunication) {
741
+ this.debug(`>>> ${rawChunk}`);
742
+ }
743
+ else {
744
+ this.debug(`>>> ${frame}`);
745
+ }
746
+ if (this.forceBinaryWSFrames && typeof rawChunk === 'string') {
747
+ rawChunk = new TextEncoder().encode(rawChunk);
748
+ }
749
+ if (typeof rawChunk !== 'string' || !this.splitLargeFrames) {
750
+ this._webSocket.send(rawChunk);
751
+ }
752
+ else {
753
+ let out = rawChunk;
754
+ while (out.length > 0) {
755
+ const chunk = out.substring(0, this.maxWebSocketChunkSize);
756
+ out = out.substring(this.maxWebSocketChunkSize);
757
+ this._webSocket.send(chunk);
758
+ this.debug(`chunk sent = ${chunk.length}, remaining = ${out.length}`);
759
+ }
760
+ }
834
761
  }
835
- this.disconnectHeaders = headers;
836
- super.deactivate();
837
- }
838
- /**
839
- * Available for backward compatibility, use [Client#publish]{@link Client#publish}.
840
- *
841
- * Send a message to a named destination. Refer to your STOMP broker documentation for types
842
- * and naming of destinations. The headers will, typically, be available to the subscriber.
843
- * However, there may be special purpose headers corresponding to your STOMP broker.
844
- *
845
- * **Deprecated**, use [Client#publish]{@link Client#publish}
846
- *
847
- * Note: Body must be String. You will need to covert the payload to string in case it is not string (e.g. JSON)
848
- *
849
- * ```javascript
850
- * client.send("/queue/test", {priority: 9}, "Hello, STOMP");
851
- *
852
- * // If you want to send a message with a body, you must also pass the headers argument.
853
- * client.send("/queue/test", {}, "Hello, STOMP");
854
- * ```
855
- *
856
- * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)
857
- */
858
- send(destination, headers = {}, body = '') {
859
- headers = Object.assign({}, headers);
860
- const skipContentLengthHeader = headers['content-length'] === false;
861
- if (skipContentLengthHeader) {
862
- delete headers['content-length'];
863
- }
864
- this.publish({
865
- destination,
866
- headers: headers,
867
- body,
868
- skipContentLengthHeader,
869
- });
870
- }
871
- /**
872
- * Available for backward compatibility, renamed to [Client#reconnectDelay]{@link Client#reconnectDelay}.
873
- *
874
- * **Deprecated**
875
- */
876
- set reconnect_delay(value) {
877
- this.reconnectDelay = value;
878
- }
879
- /**
880
- * Available for backward compatibility, renamed to [Client#webSocket]{@link Client#webSocket}.
881
- *
882
- * **Deprecated**
883
- */
884
- get ws() {
885
- return this.webSocket;
886
- }
887
- /**
888
- * Available for backward compatibility, renamed to [Client#connectedVersion]{@link Client#connectedVersion}.
889
- *
890
- * **Deprecated**
891
- */
892
- get version() {
893
- return this.connectedVersion;
894
- }
895
- /**
896
- * Available for backward compatibility, renamed to [Client#onUnhandledMessage]{@link Client#onUnhandledMessage}.
897
- *
898
- * **Deprecated**
899
- */
900
- get onreceive() {
901
- return this.onUnhandledMessage;
902
- }
903
- /**
904
- * Available for backward compatibility, renamed to [Client#onUnhandledMessage]{@link Client#onUnhandledMessage}.
905
- *
906
- * **Deprecated**
907
- */
908
- set onreceive(value) {
909
- this.onUnhandledMessage = value;
910
- }
911
- /**
912
- * Available for backward compatibility, renamed to [Client#onUnhandledReceipt]{@link Client#onUnhandledReceipt}.
913
- * Prefer using [Client#watchForReceipt]{@link Client#watchForReceipt}.
914
- *
915
- * **Deprecated**
916
- */
917
- get onreceipt() {
918
- return this.onUnhandledReceipt;
919
- }
920
- /**
921
- * Available for backward compatibility, renamed to [Client#onUnhandledReceipt]{@link Client#onUnhandledReceipt}.
922
- *
923
- * **Deprecated**
924
- */
925
- set onreceipt(value) {
926
- this.onUnhandledReceipt = value;
927
- }
928
- /**
929
- * Available for backward compatibility, renamed to [Client#heartbeatIncoming]{@link Client#heartbeatIncoming}
930
- * [Client#heartbeatOutgoing]{@link Client#heartbeatOutgoing}.
931
- *
932
- * **Deprecated**
933
- */
934
- get heartbeat() {
935
- return this._heartbeatInfo;
936
- }
937
- /**
938
- * Available for backward compatibility, renamed to [Client#heartbeatIncoming]{@link Client#heartbeatIncoming}
939
- * [Client#heartbeatOutgoing]{@link Client#heartbeatOutgoing}.
940
- *
941
- * **Deprecated**
942
- */
943
- set heartbeat(value) {
944
- this.heartbeatIncoming = value.incoming;
945
- this.heartbeatOutgoing = value.outgoing;
946
- }
947
- }
948
-
949
-
950
- /***/ }),
951
-
952
- /***/ "./src/compatibility/heartbeat-info.ts":
953
- /*!*********************************************!*\
954
- !*** ./src/compatibility/heartbeat-info.ts ***!
955
- \*********************************************/
956
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
957
-
958
- __webpack_require__.r(__webpack_exports__);
959
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
960
- /* harmony export */ "HeartbeatInfo": () => (/* binding */ HeartbeatInfo)
961
- /* harmony export */ });
962
- /**
963
- * Part of `@stomp/stompjs`.
964
- *
965
- * @internal
966
- */
967
- class HeartbeatInfo {
968
- constructor(client) {
969
- this.client = client;
970
- }
971
- get outgoing() {
972
- return this.client.heartbeatOutgoing;
973
- }
974
- set outgoing(value) {
975
- this.client.heartbeatOutgoing = value;
976
- }
977
- get incoming() {
978
- return this.client.heartbeatIncoming;
979
- }
980
- set incoming(value) {
981
- this.client.heartbeatIncoming = value;
982
- }
983
- }
984
-
985
-
986
- /***/ }),
987
-
988
- /***/ "./src/compatibility/stomp.ts":
989
- /*!************************************!*\
990
- !*** ./src/compatibility/stomp.ts ***!
991
- \************************************/
992
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
993
-
994
- __webpack_require__.r(__webpack_exports__);
995
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
996
- /* harmony export */ "Stomp": () => (/* binding */ Stomp)
997
- /* harmony export */ });
998
- /* harmony import */ var _versions__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../versions */ "./src/versions.ts");
999
- /* harmony import */ var _compat_client__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./compat-client */ "./src/compatibility/compat-client.ts");
1000
-
1001
-
1002
- /**
1003
- * STOMP Class, acts like a factory to create {@link Client}.
1004
- *
1005
- * Part of `@stomp/stompjs`.
1006
- *
1007
- * **Deprecated**
1008
- *
1009
- * It will be removed in next major version. Please switch to {@link Client}.
1010
- */
1011
- class Stomp {
1012
- /**
1013
- * This method creates a WebSocket client that is connected to
1014
- * the STOMP server located at the url.
1015
- *
1016
- * ```javascript
1017
- * var url = "ws://localhost:61614/stomp";
1018
- * var client = Stomp.client(url);
1019
- * ```
1020
- *
1021
- * **Deprecated**
1022
- *
1023
- * It will be removed in next major version. Please switch to {@link Client}
1024
- * using [Client#brokerURL]{@link Client#brokerURL}.
1025
- */
1026
- static client(url, protocols) {
1027
- // This is a hack to allow another implementation than the standard
1028
- // HTML5 WebSocket class.
1029
- //
1030
- // It is possible to use another class by calling
1031
- //
1032
- // Stomp.WebSocketClass = MozWebSocket
1033
- //
1034
- // *prior* to call `Stomp.client()`.
1035
- //
1036
- // This hack is deprecated and `Stomp.over()` method should be used
1037
- // instead.
1038
- // See remarks on the function Stomp.over
1039
- if (protocols == null) {
1040
- protocols = _versions__WEBPACK_IMPORTED_MODULE_0__.Versions["default"].protocolVersions();
1041
- }
1042
- const wsFn = () => {
1043
- const klass = Stomp.WebSocketClass || WebSocket;
1044
- return new klass(url, protocols);
1045
- };
1046
- return new _compat_client__WEBPACK_IMPORTED_MODULE_1__.CompatClient(wsFn);
1047
- }
1048
- /**
1049
- * This method is an alternative to [Stomp#client]{@link Stomp#client} to let the user
1050
- * specify the WebSocket to use (either a standard HTML5 WebSocket or
1051
- * a similar object).
1052
- *
1053
- * In order to support reconnection, the function Client._connect should be callable more than once.
1054
- * While reconnecting
1055
- * a new instance of underlying transport (TCP Socket, WebSocket or SockJS) will be needed. So, this function
1056
- * alternatively allows passing a function that should return a new instance of the underlying socket.
1057
- *
1058
- * ```javascript
1059
- * var client = Stomp.over(function(){
1060
- * return new WebSocket('ws://localhost:15674/ws')
1061
- * });
1062
- * ```
1063
- *
1064
- * **Deprecated**
1065
- *
1066
- * It will be removed in next major version. Please switch to {@link Client}
1067
- * using [Client#webSocketFactory]{@link Client#webSocketFactory}.
1068
- */
1069
- static over(ws) {
1070
- let wsFn;
1071
- if (typeof ws === 'function') {
1072
- wsFn = ws;
762
+ dispose() {
763
+ if (this.connected) {
764
+ try {
765
+ // clone before updating
766
+ const disconnectHeaders = Object.assign({}, this.disconnectHeaders);
767
+ if (!disconnectHeaders.receipt) {
768
+ disconnectHeaders.receipt = `close-${this._counter++}`;
769
+ }
770
+ this.watchForReceipt(disconnectHeaders.receipt, frame => {
771
+ this._closeWebsocket();
772
+ this._cleanUp();
773
+ this.onDisconnect(frame);
774
+ });
775
+ this._transmit({ command: 'DISCONNECT', headers: disconnectHeaders });
776
+ }
777
+ catch (error) {
778
+ this.debug(`Ignoring error during disconnect ${error}`);
779
+ }
780
+ }
781
+ else {
782
+ if (this._webSocket.readyState === exports.StompSocketState.CONNECTING ||
783
+ this._webSocket.readyState === exports.StompSocketState.OPEN) {
784
+ this._closeWebsocket();
785
+ }
786
+ }
787
+ }
788
+ _cleanUp() {
789
+ this._connected = false;
790
+ if (this._pinger) {
791
+ clearInterval(this._pinger);
792
+ this._pinger = undefined;
793
+ }
794
+ if (this._ponger) {
795
+ clearInterval(this._ponger);
796
+ this._ponger = undefined;
797
+ }
798
+ }
799
+ publish(params) {
800
+ const { destination, headers, body, binaryBody, skipContentLengthHeader } = params;
801
+ const hdrs = Object.assign({ destination }, headers);
802
+ this._transmit({
803
+ command: 'SEND',
804
+ headers: hdrs,
805
+ body,
806
+ binaryBody,
807
+ skipContentLengthHeader,
808
+ });
809
+ }
810
+ watchForReceipt(receiptId, callback) {
811
+ this._receiptWatchers[receiptId] = callback;
1073
812
  }
1074
- else {
1075
- console.warn('Stomp.over did not receive a factory, auto reconnect will not work. ' +
1076
- 'Please see https://stomp-js.github.io/api-docs/latest/classes/Stomp.html#over');
1077
- wsFn = () => ws;
813
+ subscribe(destination, callback, headers = {}) {
814
+ headers = Object.assign({}, headers);
815
+ if (!headers.id) {
816
+ headers.id = `sub-${this._counter++}`;
817
+ }
818
+ headers.destination = destination;
819
+ this._subscriptions[headers.id] = callback;
820
+ this._transmit({ command: 'SUBSCRIBE', headers });
821
+ const client = this;
822
+ return {
823
+ id: headers.id,
824
+ unsubscribe(hdrs) {
825
+ return client.unsubscribe(headers.id, hdrs);
826
+ },
827
+ };
828
+ }
829
+ unsubscribe(id, headers = {}) {
830
+ headers = Object.assign({}, headers);
831
+ delete this._subscriptions[id];
832
+ headers.id = id;
833
+ this._transmit({ command: 'UNSUBSCRIBE', headers });
834
+ }
835
+ begin(transactionId) {
836
+ const txId = transactionId || `tx-${this._counter++}`;
837
+ this._transmit({
838
+ command: 'BEGIN',
839
+ headers: {
840
+ transaction: txId,
841
+ },
842
+ });
843
+ const client = this;
844
+ return {
845
+ id: txId,
846
+ commit() {
847
+ client.commit(txId);
848
+ },
849
+ abort() {
850
+ client.abort(txId);
851
+ },
852
+ };
853
+ }
854
+ commit(transactionId) {
855
+ this._transmit({
856
+ command: 'COMMIT',
857
+ headers: {
858
+ transaction: transactionId,
859
+ },
860
+ });
861
+ }
862
+ abort(transactionId) {
863
+ this._transmit({
864
+ command: 'ABORT',
865
+ headers: {
866
+ transaction: transactionId,
867
+ },
868
+ });
869
+ }
870
+ ack(messageId, subscriptionId, headers = {}) {
871
+ headers = Object.assign({}, headers);
872
+ if (this._connectedVersion === Versions.V1_2) {
873
+ headers.id = messageId;
874
+ }
875
+ else {
876
+ headers['message-id'] = messageId;
877
+ }
878
+ headers.subscription = subscriptionId;
879
+ this._transmit({ command: 'ACK', headers });
880
+ }
881
+ nack(messageId, subscriptionId, headers = {}) {
882
+ headers = Object.assign({}, headers);
883
+ if (this._connectedVersion === Versions.V1_2) {
884
+ headers.id = messageId;
885
+ }
886
+ else {
887
+ headers['message-id'] = messageId;
888
+ }
889
+ headers.subscription = subscriptionId;
890
+ return this._transmit({ command: 'NACK', headers });
1078
891
  }
1079
- return new _compat_client__WEBPACK_IMPORTED_MODULE_1__.CompatClient(wsFn);
1080
892
  }
1081
- }
1082
- /**
1083
- * In case you need to use a non standard class for WebSocket.
1084
- *
1085
- * For example when using within NodeJS environment:
1086
- *
1087
- * ```javascript
1088
- * StompJs = require('../../esm5/');
1089
- * Stomp = StompJs.Stomp;
1090
- * Stomp.WebSocketClass = require('websocket').w3cwebsocket;
1091
- * ```
1092
- *
1093
- * **Deprecated**
1094
- *
1095
- *
1096
- * It will be removed in next major version. Please switch to {@link Client}
1097
- * using [Client#webSocketFactory]{@link Client#webSocketFactory}.
1098
- */
1099
- // tslint:disable-next-line:variable-name
1100
- Stomp.WebSocketClass = null;
1101
-
1102
893
 
1103
- /***/ }),
1104
-
1105
- /***/ "./src/frame-impl.ts":
1106
- /*!***************************!*\
1107
- !*** ./src/frame-impl.ts ***!
1108
- \***************************/
1109
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1110
-
1111
- __webpack_require__.r(__webpack_exports__);
1112
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1113
- /* harmony export */ "FrameImpl": () => (/* binding */ FrameImpl)
1114
- /* harmony export */ });
1115
- /* harmony import */ var _byte__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./byte */ "./src/byte.ts");
1116
-
1117
- /**
1118
- * Frame class represents a STOMP frame.
1119
- *
1120
- * @internal
1121
- */
1122
- class FrameImpl {
1123
894
  /**
1124
- * Frame constructor. `command`, `headers` and `body` are available as properties.
895
+ * STOMP Client Class.
1125
896
  *
1126
- * @internal
897
+ * Part of `@stomp/stompjs`.
1127
898
  */
1128
- constructor(params) {
1129
- const { command, headers, body, binaryBody, escapeHeaderValues, skipContentLengthHeader, } = params;
1130
- this.command = command;
1131
- this.headers = Object.assign({}, headers || {});
1132
- if (binaryBody) {
1133
- this._binaryBody = binaryBody;
1134
- this.isBinaryBody = true;
1135
- }
1136
- else {
1137
- this._body = body || '';
1138
- this.isBinaryBody = false;
1139
- }
1140
- this.escapeHeaderValues = escapeHeaderValues || false;
1141
- this.skipContentLengthHeader = skipContentLengthHeader || false;
1142
- }
1143
- /**
1144
- * body of the frame
1145
- */
1146
- get body() {
1147
- if (!this._body && this.isBinaryBody) {
1148
- this._body = new TextDecoder().decode(this._binaryBody);
899
+ class Client {
900
+ /**
901
+ * Create an instance.
902
+ */
903
+ constructor(conf = {}) {
904
+ /**
905
+ * STOMP versions to attempt during STOMP handshake. By default versions `1.0`, `1.1`, and `1.2` are attempted.
906
+ *
907
+ * Example:
908
+ * ```javascript
909
+ * // Try only versions 1.0 and 1.1
910
+ * client.stompVersions = new Versions(['1.0', '1.1'])
911
+ * ```
912
+ */
913
+ this.stompVersions = Versions.default;
914
+ /**
915
+ * Will retry if Stomp connection is not established in specified milliseconds.
916
+ * Default 0, which implies wait for ever.
917
+ */
918
+ this.connectionTimeout = 0;
919
+ /**
920
+ * automatically reconnect with delay in milliseconds, set to 0 to disable.
921
+ */
922
+ this.reconnectDelay = 5000;
923
+ /**
924
+ * Incoming heartbeat interval in milliseconds. Set to 0 to disable.
925
+ */
926
+ this.heartbeatIncoming = 10000;
927
+ /**
928
+ * Outgoing heartbeat interval in milliseconds. Set to 0 to disable.
929
+ */
930
+ this.heartbeatOutgoing = 10000;
931
+ /**
932
+ * This switches on a non standard behavior while sending WebSocket packets.
933
+ * It splits larger (text) packets into chunks of [maxWebSocketChunkSize]{@link Client#maxWebSocketChunkSize}.
934
+ * Only Java Spring brokers seems to use this mode.
935
+ *
936
+ * WebSockets, by itself, split large (text) packets,
937
+ * so it is not needed with a truly compliant STOMP/WebSocket broker.
938
+ * Actually setting it for such broker will cause large messages to fail.
939
+ *
940
+ * `false` by default.
941
+ *
942
+ * Binary frames are never split.
943
+ */
944
+ this.splitLargeFrames = false;
945
+ /**
946
+ * See [splitLargeFrames]{@link Client#splitLargeFrames}.
947
+ * This has no effect if [splitLargeFrames]{@link Client#splitLargeFrames} is `false`.
948
+ */
949
+ this.maxWebSocketChunkSize = 8 * 1024;
950
+ /**
951
+ * Usually the
952
+ * [type of WebSocket frame]{@link https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/send#Parameters}
953
+ * is automatically decided by type of the payload.
954
+ * Default is `false`, which should work with all compliant brokers.
955
+ *
956
+ * Set this flag to force binary frames.
957
+ */
958
+ this.forceBinaryWSFrames = false;
959
+ /**
960
+ * A bug in ReactNative chops a string on occurrence of a NULL.
961
+ * See issue [https://github.com/stomp-js/stompjs/issues/89]{@link https://github.com/stomp-js/stompjs/issues/89}.
962
+ * This makes incoming WebSocket messages invalid STOMP packets.
963
+ * Setting this flag attempts to reverse the damage by appending a NULL.
964
+ * If the broker splits a large message into multiple WebSocket messages,
965
+ * this flag will cause data loss and abnormal termination of connection.
966
+ *
967
+ * This is not an ideal solution, but a stop gap until the underlying issue is fixed at ReactNative library.
968
+ */
969
+ this.appendMissingNULLonIncoming = false;
970
+ /**
971
+ * Browsers do not immediately close WebSockets when `.close` is issued.
972
+ * This may cause reconnection to take a longer on certain type of failures.
973
+ * In case of incoming heartbeat failure, this experimental flag instructs the library
974
+ * to discard the socket immediately (even before it is actually closed).
975
+ */
976
+ this.discardWebsocketOnCommFailure = false;
977
+ /**
978
+ * Activation state.
979
+ *
980
+ * It will usually be ACTIVE or INACTIVE.
981
+ * When deactivating it may go from ACTIVE to INACTIVE without entering DEACTIVATING.
982
+ */
983
+ this.state = exports.ActivationState.INACTIVE;
984
+ // Dummy callbacks
985
+ const noOp = () => { };
986
+ this.debug = noOp;
987
+ this.beforeConnect = noOp;
988
+ this.onConnect = noOp;
989
+ this.onDisconnect = noOp;
990
+ this.onUnhandledMessage = noOp;
991
+ this.onUnhandledReceipt = noOp;
992
+ this.onUnhandledFrame = noOp;
993
+ this.onStompError = noOp;
994
+ this.onWebSocketClose = noOp;
995
+ this.onWebSocketError = noOp;
996
+ this.logRawCommunication = false;
997
+ this.onChangeState = noOp;
998
+ // These parameters would typically get proper values before connect is called
999
+ this.connectHeaders = {};
1000
+ this._disconnectHeaders = {};
1001
+ // Apply configuration
1002
+ this.configure(conf);
1149
1003
  }
1150
- return this._body || '';
1151
- }
1152
- /**
1153
- * body as Uint8Array
1154
- */
1155
- get binaryBody() {
1156
- if (!this._binaryBody && !this.isBinaryBody) {
1157
- this._binaryBody = new TextEncoder().encode(this._body);
1004
+ /**
1005
+ * Underlying WebSocket instance, READONLY.
1006
+ */
1007
+ get webSocket() {
1008
+ return this._stompHandler?._webSocket;
1158
1009
  }
1159
- // At this stage it will definitely have a valid value
1160
- return this._binaryBody;
1161
- }
1162
- /**
1163
- * deserialize a STOMP Frame from raw data.
1164
- *
1165
- * @internal
1166
- */
1167
- static fromRawFrame(rawFrame, escapeHeaderValues) {
1168
- const headers = {};
1169
- const trim = (str) => str.replace(/^\s+|\s+$/g, '');
1170
- // In case of repeated headers, as per standards, first value need to be used
1171
- for (const header of rawFrame.headers.reverse()) {
1172
- const idx = header.indexOf(':');
1173
- const key = trim(header[0]);
1174
- let value = trim(header[1]);
1175
- if (escapeHeaderValues &&
1176
- rawFrame.command !== 'CONNECT' &&
1177
- rawFrame.command !== 'CONNECTED') {
1178
- value = FrameImpl.hdrValueUnEscape(value);
1179
- }
1180
- headers[key] = value;
1181
- }
1182
- return new FrameImpl({
1183
- command: rawFrame.command,
1184
- headers,
1185
- binaryBody: rawFrame.binaryBody,
1186
- escapeHeaderValues,
1187
- });
1188
- }
1189
- /**
1190
- * @internal
1191
- */
1192
- toString() {
1193
- return this.serializeCmdAndHeaders();
1194
- }
1195
- /**
1196
- * serialize this Frame in a format suitable to be passed to WebSocket.
1197
- * If the body is string the output will be string.
1198
- * If the body is binary (i.e. of type Unit8Array) it will be serialized to ArrayBuffer.
1199
- *
1200
- * @internal
1201
- */
1202
- serialize() {
1203
- const cmdAndHeaders = this.serializeCmdAndHeaders();
1204
- if (this.isBinaryBody) {
1205
- return FrameImpl.toUnit8Array(cmdAndHeaders, this._binaryBody).buffer;
1010
+ /**
1011
+ * Disconnection headers.
1012
+ */
1013
+ get disconnectHeaders() {
1014
+ return this._disconnectHeaders;
1015
+ }
1016
+ set disconnectHeaders(value) {
1017
+ this._disconnectHeaders = value;
1018
+ if (this._stompHandler) {
1019
+ this._stompHandler.disconnectHeaders = this._disconnectHeaders;
1020
+ }
1206
1021
  }
1207
- else {
1208
- return cmdAndHeaders + this._body + _byte__WEBPACK_IMPORTED_MODULE_0__.BYTE.NULL;
1022
+ /**
1023
+ * `true` if there is a active connection with STOMP Broker
1024
+ */
1025
+ get connected() {
1026
+ return !!this._stompHandler && this._stompHandler.connected;
1027
+ }
1028
+ /**
1029
+ * version of STOMP protocol negotiated with the server, READONLY
1030
+ */
1031
+ get connectedVersion() {
1032
+ return this._stompHandler ? this._stompHandler.connectedVersion : undefined;
1033
+ }
1034
+ /**
1035
+ * if the client is active (connected or going to reconnect)
1036
+ */
1037
+ get active() {
1038
+ return this.state === exports.ActivationState.ACTIVE;
1039
+ }
1040
+ _changeState(state) {
1041
+ this.state = state;
1042
+ this.onChangeState(state);
1043
+ }
1044
+ /**
1045
+ * Update configuration.
1046
+ */
1047
+ configure(conf) {
1048
+ // bulk assign all properties to this
1049
+ Object.assign(this, conf);
1050
+ }
1051
+ /**
1052
+ * Initiate the connection with the broker.
1053
+ * If the connection breaks, as per [Client#reconnectDelay]{@link Client#reconnectDelay},
1054
+ * it will keep trying to reconnect.
1055
+ *
1056
+ * Call [Client#deactivate]{@link Client#deactivate} to disconnect and stop reconnection attempts.
1057
+ */
1058
+ activate() {
1059
+ if (this.state === exports.ActivationState.DEACTIVATING) {
1060
+ this.debug('Still DEACTIVATING, please await call to deactivate before trying to re-activate');
1061
+ throw new Error('Still DEACTIVATING, can not activate now');
1062
+ }
1063
+ if (this.active) {
1064
+ this.debug('Already ACTIVE, ignoring request to activate');
1065
+ return;
1066
+ }
1067
+ this._changeState(exports.ActivationState.ACTIVE);
1068
+ this._connect();
1069
+ }
1070
+ async _connect() {
1071
+ if (this.connected) {
1072
+ this.debug('STOMP: already connected, nothing to do');
1073
+ return;
1074
+ }
1075
+ await this.beforeConnect();
1076
+ if (!this.active) {
1077
+ this.debug('Client has been marked inactive, will not attempt to connect');
1078
+ return;
1079
+ }
1080
+ // setup connection watcher
1081
+ if (this.connectionTimeout > 0) {
1082
+ // clear first
1083
+ if (this._connectionWatcher) {
1084
+ clearTimeout(this._connectionWatcher);
1085
+ }
1086
+ this._connectionWatcher = setTimeout(() => {
1087
+ if (this.connected) {
1088
+ return;
1089
+ }
1090
+ // Connection not established, close the underlying socket
1091
+ // a reconnection will be attempted
1092
+ this.debug(`Connection not established in ${this.connectionTimeout}ms, closing socket`);
1093
+ this.forceDisconnect();
1094
+ }, this.connectionTimeout);
1095
+ }
1096
+ this.debug('Opening Web Socket...');
1097
+ // Get the actual WebSocket (or a similar object)
1098
+ const webSocket = this._createWebSocket();
1099
+ this._stompHandler = new StompHandler(this, webSocket, {
1100
+ debug: this.debug,
1101
+ stompVersions: this.stompVersions,
1102
+ connectHeaders: this.connectHeaders,
1103
+ disconnectHeaders: this._disconnectHeaders,
1104
+ heartbeatIncoming: this.heartbeatIncoming,
1105
+ heartbeatOutgoing: this.heartbeatOutgoing,
1106
+ splitLargeFrames: this.splitLargeFrames,
1107
+ maxWebSocketChunkSize: this.maxWebSocketChunkSize,
1108
+ forceBinaryWSFrames: this.forceBinaryWSFrames,
1109
+ logRawCommunication: this.logRawCommunication,
1110
+ appendMissingNULLonIncoming: this.appendMissingNULLonIncoming,
1111
+ discardWebsocketOnCommFailure: this.discardWebsocketOnCommFailure,
1112
+ onConnect: frame => {
1113
+ // Successfully connected, stop the connection watcher
1114
+ if (this._connectionWatcher) {
1115
+ clearTimeout(this._connectionWatcher);
1116
+ this._connectionWatcher = undefined;
1117
+ }
1118
+ if (!this.active) {
1119
+ this.debug('STOMP got connected while deactivate was issued, will disconnect now');
1120
+ this._disposeStompHandler();
1121
+ return;
1122
+ }
1123
+ this.onConnect(frame);
1124
+ },
1125
+ onDisconnect: frame => {
1126
+ this.onDisconnect(frame);
1127
+ },
1128
+ onStompError: frame => {
1129
+ this.onStompError(frame);
1130
+ },
1131
+ onWebSocketClose: evt => {
1132
+ this._stompHandler = undefined; // a new one will be created in case of a reconnect
1133
+ if (this.state === exports.ActivationState.DEACTIVATING) {
1134
+ // Mark deactivation complete
1135
+ this._changeState(exports.ActivationState.INACTIVE);
1136
+ }
1137
+ // The callback is called before attempting to reconnect, this would allow the client
1138
+ // to be `deactivated` in the callback.
1139
+ this.onWebSocketClose(evt);
1140
+ if (this.active) {
1141
+ this._schedule_reconnect();
1142
+ }
1143
+ },
1144
+ onWebSocketError: evt => {
1145
+ this.onWebSocketError(evt);
1146
+ },
1147
+ onUnhandledMessage: message => {
1148
+ this.onUnhandledMessage(message);
1149
+ },
1150
+ onUnhandledReceipt: frame => {
1151
+ this.onUnhandledReceipt(frame);
1152
+ },
1153
+ onUnhandledFrame: frame => {
1154
+ this.onUnhandledFrame(frame);
1155
+ },
1156
+ });
1157
+ this._stompHandler.start();
1158
+ }
1159
+ _createWebSocket() {
1160
+ let webSocket;
1161
+ if (this.webSocketFactory) {
1162
+ webSocket = this.webSocketFactory();
1163
+ }
1164
+ else if (this.brokerURL) {
1165
+ webSocket = new WebSocket(this.brokerURL, this.stompVersions.protocolVersions());
1166
+ }
1167
+ else {
1168
+ throw new Error('Either brokerURL or webSocketFactory must be provided');
1169
+ }
1170
+ webSocket.binaryType = 'arraybuffer';
1171
+ return webSocket;
1172
+ }
1173
+ _schedule_reconnect() {
1174
+ if (this.reconnectDelay > 0) {
1175
+ this.debug(`STOMP: scheduling reconnection in ${this.reconnectDelay}ms`);
1176
+ this._reconnector = setTimeout(() => {
1177
+ this._connect();
1178
+ }, this.reconnectDelay);
1179
+ }
1180
+ }
1181
+ /**
1182
+ * Disconnect if connected and stop auto reconnect loop.
1183
+ * Appropriate callbacks will be invoked if there is an underlying STOMP connection.
1184
+ *
1185
+ * This call is async. It will resolve immediately if there is no underlying active websocket,
1186
+ * otherwise, it will resolve after the underlying websocket is properly disposed of.
1187
+ *
1188
+ * It is not an error to invoke this method more than once.
1189
+ * Each of those would resolve on completion of deactivation.
1190
+ *
1191
+ * To reactivate, you can call [Client#activate]{@link Client#activate}.
1192
+ *
1193
+ * Experimental: pass `force: true` to immediately discard the underlying connection.
1194
+ * This mode will skip both the STOMP and the Websocket shutdown sequences.
1195
+ * In some cases, browsers take a long time in the Websocket shutdown if the underlying connection had gone stale.
1196
+ * Using this mode can speed up.
1197
+ * When this mode is used, the actual Websocket may linger for a while
1198
+ * and the broker may not realize that the connection is no longer in use.
1199
+ *
1200
+ * It is possible to invoke this method initially without the `force` option
1201
+ * and subsequently, say after a wait, with the `force` option.
1202
+ */
1203
+ async deactivate(options = {}) {
1204
+ const force = options.force || false;
1205
+ const needToDispose = this.active;
1206
+ let retPromise;
1207
+ if (this.state === exports.ActivationState.INACTIVE) {
1208
+ this.debug(`Already INACTIVE, nothing more to do`);
1209
+ return Promise.resolve();
1210
+ }
1211
+ this._changeState(exports.ActivationState.DEACTIVATING);
1212
+ // Clear if a reconnection was scheduled
1213
+ if (this._reconnector) {
1214
+ clearTimeout(this._reconnector);
1215
+ this._reconnector = undefined;
1216
+ }
1217
+ if (this._stompHandler &&
1218
+ // @ts-ignore - if there is a _stompHandler, there is the webSocket
1219
+ this.webSocket.readyState !== exports.StompSocketState.CLOSED) {
1220
+ const origOnWebSocketClose = this._stompHandler.onWebSocketClose;
1221
+ // we need to wait for the underlying websocket to close
1222
+ retPromise = new Promise((resolve, reject) => {
1223
+ // @ts-ignore - there is a _stompHandler
1224
+ this._stompHandler.onWebSocketClose = evt => {
1225
+ origOnWebSocketClose(evt);
1226
+ resolve();
1227
+ };
1228
+ });
1229
+ }
1230
+ else {
1231
+ // indicate that auto reconnect loop should terminate
1232
+ this._changeState(exports.ActivationState.INACTIVE);
1233
+ return Promise.resolve();
1234
+ }
1235
+ if (force) {
1236
+ this._stompHandler?.discardWebsocket();
1237
+ }
1238
+ else if (needToDispose) {
1239
+ this._disposeStompHandler();
1240
+ }
1241
+ return retPromise;
1242
+ }
1243
+ /**
1244
+ * Force disconnect if there is an active connection by directly closing the underlying WebSocket.
1245
+ * This is different than a normal disconnect where a DISCONNECT sequence is carried out with the broker.
1246
+ * After forcing disconnect, automatic reconnect will be attempted.
1247
+ * To stop further reconnects call [Client#deactivate]{@link Client#deactivate} as well.
1248
+ */
1249
+ forceDisconnect() {
1250
+ if (this._stompHandler) {
1251
+ this._stompHandler.forceDisconnect();
1252
+ }
1253
+ }
1254
+ _disposeStompHandler() {
1255
+ // Dispose STOMP Handler
1256
+ if (this._stompHandler) {
1257
+ this._stompHandler.dispose();
1258
+ }
1259
+ }
1260
+ /**
1261
+ * Send a message to a named destination. Refer to your STOMP broker documentation for types
1262
+ * and naming of destinations.
1263
+ *
1264
+ * STOMP protocol specifies and suggests some headers and also allows broker specific headers.
1265
+ *
1266
+ * `body` must be String.
1267
+ * You will need to covert the payload to string in case it is not string (e.g. JSON).
1268
+ *
1269
+ * To send a binary message body use binaryBody parameter. It should be a
1270
+ * [Uint8Array](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Uint8Array).
1271
+ * Sometimes brokers may not support binary frames out of the box.
1272
+ * Please check your broker documentation.
1273
+ *
1274
+ * `content-length` header is automatically added to the STOMP Frame sent to the broker.
1275
+ * Set `skipContentLengthHeader` to indicate that `content-length` header should not be added.
1276
+ * For binary messages `content-length` header is always added.
1277
+ *
1278
+ * Caution: The broker will, most likely, report an error and disconnect if message body has NULL octet(s)
1279
+ * and `content-length` header is missing.
1280
+ *
1281
+ * ```javascript
1282
+ * client.publish({destination: "/queue/test", headers: {priority: 9}, body: "Hello, STOMP"});
1283
+ *
1284
+ * // Only destination is mandatory parameter
1285
+ * client.publish({destination: "/queue/test", body: "Hello, STOMP"});
1286
+ *
1287
+ * // Skip content-length header in the frame to the broker
1288
+ * client.publish({"/queue/test", body: "Hello, STOMP", skipContentLengthHeader: true});
1289
+ *
1290
+ * var binaryData = generateBinaryData(); // This need to be of type Uint8Array
1291
+ * // setting content-type header is not mandatory, however a good practice
1292
+ * client.publish({destination: '/topic/special', binaryBody: binaryData,
1293
+ * headers: {'content-type': 'application/octet-stream'}});
1294
+ * ```
1295
+ */
1296
+ publish(params) {
1297
+ this._checkConnection();
1298
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1299
+ this._stompHandler.publish(params);
1300
+ }
1301
+ _checkConnection() {
1302
+ if (!this.connected) {
1303
+ throw new TypeError('There is no underlying STOMP connection');
1304
+ }
1305
+ }
1306
+ /**
1307
+ * STOMP brokers may carry out operation asynchronously and allow requesting for acknowledgement.
1308
+ * To request an acknowledgement, a `receipt` header needs to be sent with the actual request.
1309
+ * The value (say receipt-id) for this header needs to be unique for each use. Typically a sequence, a UUID, a
1310
+ * random number or a combination may be used.
1311
+ *
1312
+ * A complaint broker will send a RECEIPT frame when an operation has actually been completed.
1313
+ * The operation needs to be matched based in the value of the receipt-id.
1314
+ *
1315
+ * This method allow watching for a receipt and invoke the callback
1316
+ * when corresponding receipt has been received.
1317
+ *
1318
+ * The actual {@link FrameImpl} will be passed as parameter to the callback.
1319
+ *
1320
+ * Example:
1321
+ * ```javascript
1322
+ * // Subscribing with acknowledgement
1323
+ * let receiptId = randomText();
1324
+ *
1325
+ * client.watchForReceipt(receiptId, function() {
1326
+ * // Will be called after server acknowledges
1327
+ * });
1328
+ *
1329
+ * client.subscribe(TEST.destination, onMessage, {receipt: receiptId});
1330
+ *
1331
+ *
1332
+ * // Publishing with acknowledgement
1333
+ * receiptId = randomText();
1334
+ *
1335
+ * client.watchForReceipt(receiptId, function() {
1336
+ * // Will be called after server acknowledges
1337
+ * });
1338
+ * client.publish({destination: TEST.destination, headers: {receipt: receiptId}, body: msg});
1339
+ * ```
1340
+ */
1341
+ watchForReceipt(receiptId, callback) {
1342
+ this._checkConnection();
1343
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1344
+ this._stompHandler.watchForReceipt(receiptId, callback);
1345
+ }
1346
+ /**
1347
+ * Subscribe to a STOMP Broker location. The callback will be invoked for each received message with
1348
+ * the {@link IMessage} as argument.
1349
+ *
1350
+ * Note: The library will generate an unique ID if there is none provided in the headers.
1351
+ * To use your own ID, pass it using the headers argument.
1352
+ *
1353
+ * ```javascript
1354
+ * callback = function(message) {
1355
+ * // called when the client receives a STOMP message from the server
1356
+ * if (message.body) {
1357
+ * alert("got message with body " + message.body)
1358
+ * } else {
1359
+ * alert("got empty message");
1360
+ * }
1361
+ * });
1362
+ *
1363
+ * var subscription = client.subscribe("/queue/test", callback);
1364
+ *
1365
+ * // Explicit subscription id
1366
+ * var mySubId = 'my-subscription-id-001';
1367
+ * var subscription = client.subscribe(destination, callback, { id: mySubId });
1368
+ * ```
1369
+ */
1370
+ subscribe(destination, callback, headers = {}) {
1371
+ this._checkConnection();
1372
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1373
+ return this._stompHandler.subscribe(destination, callback, headers);
1374
+ }
1375
+ /**
1376
+ * It is preferable to unsubscribe from a subscription by calling
1377
+ * `unsubscribe()` directly on {@link StompSubscription} returned by `client.subscribe()`:
1378
+ *
1379
+ * ```javascript
1380
+ * var subscription = client.subscribe(destination, onmessage);
1381
+ * // ...
1382
+ * subscription.unsubscribe();
1383
+ * ```
1384
+ *
1385
+ * See: http://stomp.github.com/stomp-specification-1.2.html#UNSUBSCRIBE UNSUBSCRIBE Frame
1386
+ */
1387
+ unsubscribe(id, headers = {}) {
1388
+ this._checkConnection();
1389
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1390
+ this._stompHandler.unsubscribe(id, headers);
1391
+ }
1392
+ /**
1393
+ * Start a transaction, the returned {@link ITransaction} has methods - [commit]{@link ITransaction#commit}
1394
+ * and [abort]{@link ITransaction#abort}.
1395
+ *
1396
+ * `transactionId` is optional, if not passed the library will generate it internally.
1397
+ */
1398
+ begin(transactionId) {
1399
+ this._checkConnection();
1400
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1401
+ return this._stompHandler.begin(transactionId);
1402
+ }
1403
+ /**
1404
+ * Commit a transaction.
1405
+ *
1406
+ * It is preferable to commit a transaction by calling [commit]{@link ITransaction#commit} directly on
1407
+ * {@link ITransaction} returned by [client.begin]{@link Client#begin}.
1408
+ *
1409
+ * ```javascript
1410
+ * var tx = client.begin(txId);
1411
+ * //...
1412
+ * tx.commit();
1413
+ * ```
1414
+ */
1415
+ commit(transactionId) {
1416
+ this._checkConnection();
1417
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1418
+ this._stompHandler.commit(transactionId);
1419
+ }
1420
+ /**
1421
+ * Abort a transaction.
1422
+ * It is preferable to abort a transaction by calling [abort]{@link ITransaction#abort} directly on
1423
+ * {@link ITransaction} returned by [client.begin]{@link Client#begin}.
1424
+ *
1425
+ * ```javascript
1426
+ * var tx = client.begin(txId);
1427
+ * //...
1428
+ * tx.abort();
1429
+ * ```
1430
+ */
1431
+ abort(transactionId) {
1432
+ this._checkConnection();
1433
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1434
+ this._stompHandler.abort(transactionId);
1209
1435
  }
1210
- }
1211
- serializeCmdAndHeaders() {
1212
- const lines = [this.command];
1213
- if (this.skipContentLengthHeader) {
1214
- delete this.headers['content-length'];
1215
- }
1216
- for (const name of Object.keys(this.headers || {})) {
1217
- const value = this.headers[name];
1218
- if (this.escapeHeaderValues &&
1219
- this.command !== 'CONNECT' &&
1220
- this.command !== 'CONNECTED') {
1221
- lines.push(`${name}:${FrameImpl.hdrValueEscape(`${value}`)}`);
1222
- }
1223
- else {
1224
- lines.push(`${name}:${value}`);
1225
- }
1436
+ /**
1437
+ * ACK a message. It is preferable to acknowledge a message by calling [ack]{@link IMessage#ack} directly
1438
+ * on the {@link IMessage} handled by a subscription callback:
1439
+ *
1440
+ * ```javascript
1441
+ * var callback = function (message) {
1442
+ * // process the message
1443
+ * // acknowledge it
1444
+ * message.ack();
1445
+ * };
1446
+ * client.subscribe(destination, callback, {'ack': 'client'});
1447
+ * ```
1448
+ */
1449
+ ack(messageId, subscriptionId, headers = {}) {
1450
+ this._checkConnection();
1451
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1452
+ this._stompHandler.ack(messageId, subscriptionId, headers);
1226
1453
  }
1227
- if (this.isBinaryBody ||
1228
- (!this.isBodyEmpty() && !this.skipContentLengthHeader)) {
1229
- lines.push(`content-length:${this.bodyLength()}`);
1454
+ /**
1455
+ * NACK a message. It is preferable to acknowledge a message by calling [nack]{@link IMessage#nack} directly
1456
+ * on the {@link IMessage} handled by a subscription callback:
1457
+ *
1458
+ * ```javascript
1459
+ * var callback = function (message) {
1460
+ * // process the message
1461
+ * // an error occurs, nack it
1462
+ * message.nack();
1463
+ * };
1464
+ * client.subscribe(destination, callback, {'ack': 'client'});
1465
+ * ```
1466
+ */
1467
+ nack(messageId, subscriptionId, headers = {}) {
1468
+ this._checkConnection();
1469
+ // @ts-ignore - we already checked that there is a _stompHandler, and it is connected
1470
+ this._stompHandler.nack(messageId, subscriptionId, headers);
1230
1471
  }
1231
- return lines.join(_byte__WEBPACK_IMPORTED_MODULE_0__.BYTE.LF) + _byte__WEBPACK_IMPORTED_MODULE_0__.BYTE.LF + _byte__WEBPACK_IMPORTED_MODULE_0__.BYTE.LF;
1232
- }
1233
- isBodyEmpty() {
1234
- return this.bodyLength() === 0;
1235
- }
1236
- bodyLength() {
1237
- const binaryBody = this.binaryBody;
1238
- return binaryBody ? binaryBody.length : 0;
1239
- }
1240
- /**
1241
- * Compute the size of a UTF-8 string by counting its number of bytes
1242
- * (and not the number of characters composing the string)
1243
- */
1244
- static sizeOfUTF8(s) {
1245
- return s ? new TextEncoder().encode(s).length : 0;
1246
- }
1247
- static toUnit8Array(cmdAndHeaders, binaryBody) {
1248
- const uint8CmdAndHeaders = new TextEncoder().encode(cmdAndHeaders);
1249
- const nullTerminator = new Uint8Array([0]);
1250
- const uint8Frame = new Uint8Array(uint8CmdAndHeaders.length + binaryBody.length + nullTerminator.length);
1251
- uint8Frame.set(uint8CmdAndHeaders);
1252
- uint8Frame.set(binaryBody, uint8CmdAndHeaders.length);
1253
- uint8Frame.set(nullTerminator, uint8CmdAndHeaders.length + binaryBody.length);
1254
- return uint8Frame;
1255
1472
  }
1473
+
1256
1474
  /**
1257
- * Serialize a STOMP frame as per STOMP standards, suitable to be sent to the STOMP broker.
1475
+ * Configuration options for STOMP Client, each key corresponds to
1476
+ * field by the same name in {@link Client}. This can be passed to
1477
+ * the constructor of {@link Client} or to [Client#configure]{@link Client#configure}.
1258
1478
  *
1259
- * @internal
1479
+ * There used to be a class with the same name in `@stomp/ng2-stompjs`, which has been replaced by
1480
+ * {@link RxStompConfig} and {@link InjectableRxStompConfig}.
1481
+ *
1482
+ * Part of `@stomp/stompjs`.
1260
1483
  */
1261
- static marshall(params) {
1262
- const frame = new FrameImpl(params);
1263
- return frame.serialize();
1484
+ class StompConfig {
1264
1485
  }
1486
+
1265
1487
  /**
1266
- * Escape header values
1488
+ * STOMP headers. Many functions calls will accept headers as parameters.
1489
+ * The headers sent by Broker will be available as [IFrame#headers]{@link IFrame#headers}.
1490
+ *
1491
+ * `key` and `value` must be valid strings.
1492
+ * In addition, `key` must not contain `CR`, `LF`, or `:`.
1493
+ *
1494
+ * Part of `@stomp/stompjs`.
1267
1495
  */
1268
- static hdrValueEscape(str) {
1269
- return str
1270
- .replace(/\\/g, '\\\\')
1271
- .replace(/\r/g, '\\r')
1272
- .replace(/\n/g, '\\n')
1273
- .replace(/:/g, '\\c');
1496
+ class StompHeaders {
1274
1497
  }
1498
+
1275
1499
  /**
1276
- * UnEscape header values
1500
+ * Part of `@stomp/stompjs`.
1501
+ *
1502
+ * @internal
1277
1503
  */
1278
- static hdrValueUnEscape(str) {
1279
- return str
1280
- .replace(/\\r/g, '\r')
1281
- .replace(/\\n/g, '\n')
1282
- .replace(/\\c/g, ':')
1283
- .replace(/\\\\/g, '\\');
1284
- }
1285
- }
1286
-
1287
-
1288
- /***/ }),
1289
-
1290
- /***/ "./src/parser.ts":
1291
- /*!***********************!*\
1292
- !*** ./src/parser.ts ***!
1293
- \***********************/
1294
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1295
-
1296
- __webpack_require__.r(__webpack_exports__);
1297
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1298
- /* harmony export */ "Parser": () => (/* binding */ Parser)
1299
- /* harmony export */ });
1300
- /**
1301
- * @internal
1302
- */
1303
- const NULL = 0;
1304
- /**
1305
- * @internal
1306
- */
1307
- const LF = 10;
1308
- /**
1309
- * @internal
1310
- */
1311
- const CR = 13;
1312
- /**
1313
- * @internal
1314
- */
1315
- const COLON = 58;
1316
- /**
1317
- * This is an evented, rec descent parser.
1318
- * A stream of Octets can be passed and whenever it recognizes
1319
- * a complete Frame or an incoming ping it will invoke the registered callbacks.
1320
- *
1321
- * All incoming Octets are fed into _onByte function.
1322
- * Depending on current state the _onByte function keeps changing.
1323
- * Depending on the state it keeps accumulating into _token and _results.
1324
- * State is indicated by current value of _onByte, all states are named as _collect.
1325
- *
1326
- * STOMP standards https://stomp.github.io/stomp-specification-1.2.html
1327
- * imply that all lengths are considered in bytes (instead of string lengths).
1328
- * So, before actual parsing, if the incoming data is String it is converted to Octets.
1329
- * This allows faithful implementation of the protocol and allows NULL Octets to be present in the body.
1330
- *
1331
- * There is no peek function on the incoming data.
1332
- * When a state change occurs based on an Octet without consuming the Octet,
1333
- * the Octet, after state change, is fed again (_reinjectByte).
1334
- * This became possible as the state change can be determined by inspecting just one Octet.
1335
- *
1336
- * There are two modes to collect the body, if content-length header is there then it by counting Octets
1337
- * otherwise it is determined by NULL terminator.
1338
- *
1339
- * Following the standards, the command and headers are converted to Strings
1340
- * and the body is returned as Octets.
1341
- * Headers are returned as an array and not as Hash - to allow multiple occurrence of an header.
1342
- *
1343
- * This parser does not use Regular Expressions as that can only operate on Strings.
1344
- *
1345
- * It handles if multiple STOMP frames are given as one chunk, a frame is split into multiple chunks, or
1346
- * any combination there of. The parser remembers its state (any partial frame) and continues when a new chunk
1347
- * is pushed.
1348
- *
1349
- * Typically the higher level function will convert headers to Hash, handle unescaping of header values
1350
- * (which is protocol version specific), and convert body to text.
1351
- *
1352
- * Check the parser.spec.js to understand cases that this parser is supposed to handle.
1353
- *
1354
- * Part of `@stomp/stompjs`.
1355
- *
1356
- * @internal
1357
- */
1358
- class Parser {
1359
- constructor(onFrame, onIncomingPing) {
1360
- this.onFrame = onFrame;
1361
- this.onIncomingPing = onIncomingPing;
1362
- this._encoder = new TextEncoder();
1363
- this._decoder = new TextDecoder();
1364
- this._token = [];
1365
- this._initState();
1366
- }
1367
- parseChunk(segment, appendMissingNULLonIncoming = false) {
1368
- let chunk;
1369
- if (segment instanceof ArrayBuffer) {
1370
- chunk = new Uint8Array(segment);
1371
- }
1372
- else {
1373
- chunk = this._encoder.encode(segment);
1374
- }
1375
- // See https://github.com/stomp-js/stompjs/issues/89
1376
- // Remove when underlying issue is fixed.
1377
- //
1378
- // Send a NULL byte, if the last byte of a Text frame was not NULL.F
1379
- if (appendMissingNULLonIncoming && chunk[chunk.length - 1] !== 0) {
1380
- const chunkWithNull = new Uint8Array(chunk.length + 1);
1381
- chunkWithNull.set(chunk, 0);
1382
- chunkWithNull[chunk.length] = 0;
1383
- chunk = chunkWithNull;
1384
- }
1385
- // tslint:disable-next-line:prefer-for-of
1386
- for (let i = 0; i < chunk.length; i++) {
1387
- const byte = chunk[i];
1388
- this._onByte(byte);
1389
- }
1390
- }
1391
- // The following implements a simple Rec Descent Parser.
1392
- // The grammar is simple and just one byte tells what should be the next state
1393
- _collectFrame(byte) {
1394
- if (byte === NULL) {
1395
- // Ignore
1396
- return;
1397
- }
1398
- if (byte === CR) {
1399
- // Ignore CR
1400
- return;
1401
- }
1402
- if (byte === LF) {
1403
- // Incoming Ping
1404
- this.onIncomingPing();
1405
- return;
1406
- }
1407
- this._onByte = this._collectCommand;
1408
- this._reinjectByte(byte);
1409
- }
1410
- _collectCommand(byte) {
1411
- if (byte === CR) {
1412
- // Ignore CR
1413
- return;
1414
- }
1415
- if (byte === LF) {
1416
- this._results.command = this._consumeTokenAsUTF8();
1417
- this._onByte = this._collectHeaders;
1418
- return;
1419
- }
1420
- this._consumeByte(byte);
1421
- }
1422
- _collectHeaders(byte) {
1423
- if (byte === CR) {
1424
- // Ignore CR
1425
- return;
1426
- }
1427
- if (byte === LF) {
1428
- this._setupCollectBody();
1429
- return;
1430
- }
1431
- this._onByte = this._collectHeaderKey;
1432
- this._reinjectByte(byte);
1433
- }
1434
- _reinjectByte(byte) {
1435
- this._onByte(byte);
1436
- }
1437
- _collectHeaderKey(byte) {
1438
- if (byte === COLON) {
1439
- this._headerKey = this._consumeTokenAsUTF8();
1440
- this._onByte = this._collectHeaderValue;
1441
- return;
1442
- }
1443
- this._consumeByte(byte);
1444
- }
1445
- _collectHeaderValue(byte) {
1446
- if (byte === CR) {
1447
- // Ignore CR
1448
- return;
1449
- }
1450
- if (byte === LF) {
1451
- this._results.headers.push([
1452
- this._headerKey,
1453
- this._consumeTokenAsUTF8(),
1454
- ]);
1455
- this._headerKey = undefined;
1456
- this._onByte = this._collectHeaders;
1457
- return;
1458
- }
1459
- this._consumeByte(byte);
1460
- }
1461
- _setupCollectBody() {
1462
- const contentLengthHeader = this._results.headers.filter((header) => {
1463
- return header[0] === 'content-length';
1464
- })[0];
1465
- if (contentLengthHeader) {
1466
- this._bodyBytesRemaining = parseInt(contentLengthHeader[1], 10);
1467
- this._onByte = this._collectBodyFixedSize;
1468
- }
1469
- else {
1470
- this._onByte = this._collectBodyNullTerminated;
1504
+ class HeartbeatInfo {
1505
+ constructor(client) {
1506
+ this.client = client;
1471
1507
  }
1472
- }
1473
- _collectBodyNullTerminated(byte) {
1474
- if (byte === NULL) {
1475
- this._retrievedBody();
1476
- return;
1508
+ get outgoing() {
1509
+ return this.client.heartbeatOutgoing;
1477
1510
  }
1478
- this._consumeByte(byte);
1479
- }
1480
- _collectBodyFixedSize(byte) {
1481
- // It is post decrement, so that we discard the trailing NULL octet
1482
- if (this._bodyBytesRemaining-- === 0) {
1483
- this._retrievedBody();
1484
- return;
1511
+ set outgoing(value) {
1512
+ this.client.heartbeatOutgoing = value;
1485
1513
  }
1486
- this._consumeByte(byte);
1487
- }
1488
- _retrievedBody() {
1489
- this._results.binaryBody = this._consumeTokenAsRaw();
1490
- try {
1491
- this.onFrame(this._results);
1514
+ get incoming() {
1515
+ return this.client.heartbeatIncoming;
1492
1516
  }
1493
- catch (e) {
1494
- console.log(`Ignoring an exception thrown by a frame handler. Original exception: `, e);
1517
+ set incoming(value) {
1518
+ this.client.heartbeatIncoming = value;
1495
1519
  }
1496
- this._initState();
1497
- }
1498
- // Rec Descent Parser helpers
1499
- _consumeByte(byte) {
1500
- this._token.push(byte);
1501
1520
  }
1502
- _consumeTokenAsUTF8() {
1503
- return this._decoder.decode(this._consumeTokenAsRaw());
1504
- }
1505
- _consumeTokenAsRaw() {
1506
- const rawResult = new Uint8Array(this._token);
1507
- this._token = [];
1508
- return rawResult;
1509
- }
1510
- _initState() {
1511
- this._results = {
1512
- command: undefined,
1513
- headers: [],
1514
- binaryBody: undefined,
1515
- };
1516
- this._token = [];
1517
- this._headerKey = undefined;
1518
- this._onByte = this._collectFrame;
1519
- }
1520
- }
1521
-
1522
-
1523
- /***/ }),
1524
-
1525
- /***/ "./src/stomp-config.ts":
1526
- /*!*****************************!*\
1527
- !*** ./src/stomp-config.ts ***!
1528
- \*****************************/
1529
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1530
-
1531
- __webpack_require__.r(__webpack_exports__);
1532
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1533
- /* harmony export */ "StompConfig": () => (/* binding */ StompConfig)
1534
- /* harmony export */ });
1535
- /**
1536
- * Configuration options for STOMP Client, each key corresponds to
1537
- * field by the same name in {@link Client}. This can be passed to
1538
- * the constructor of {@link Client} or to [Client#configure]{@link Client#configure}.
1539
- *
1540
- * There used to be a class with the same name in `@stomp/ng2-stompjs`, which has been replaced by
1541
- * {@link RxStompConfig} and {@link InjectableRxStompConfig}.
1542
- *
1543
- * Part of `@stomp/stompjs`.
1544
- */
1545
- class StompConfig {
1546
- }
1547
1521
 
1548
-
1549
- /***/ }),
1550
-
1551
- /***/ "./src/stomp-handler.ts":
1552
- /*!******************************!*\
1553
- !*** ./src/stomp-handler.ts ***!
1554
- \******************************/
1555
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1556
-
1557
- __webpack_require__.r(__webpack_exports__);
1558
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1559
- /* harmony export */ "StompHandler": () => (/* binding */ StompHandler)
1560
- /* harmony export */ });
1561
- /* harmony import */ var _byte__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./byte */ "./src/byte.ts");
1562
- /* harmony import */ var _frame_impl__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./frame-impl */ "./src/frame-impl.ts");
1563
- /* harmony import */ var _parser__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./parser */ "./src/parser.ts");
1564
- /* harmony import */ var _types__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./types */ "./src/types.ts");
1565
- /* harmony import */ var _versions__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./versions */ "./src/versions.ts");
1566
- /* harmony import */ var _augment_websocket__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./augment-websocket */ "./src/augment-websocket.ts");
1567
-
1568
-
1569
-
1570
-
1571
-
1572
-
1573
- /**
1574
- * The STOMP protocol handler
1575
- *
1576
- * Part of `@stomp/stompjs`.
1577
- *
1578
- * @internal
1579
- */
1580
- class StompHandler {
1581
- constructor(_client, _webSocket, config) {
1582
- this._client = _client;
1583
- this._webSocket = _webSocket;
1584
- this._connected = false;
1585
- this._serverFrameHandlers = {
1586
- // [CONNECTED Frame](http://stomp.github.com/stomp-specification-1.2.html#CONNECTED_Frame)
1587
- CONNECTED: frame => {
1588
- this.debug(`connected to server ${frame.headers.server}`);
1589
- this._connected = true;
1590
- this._connectedVersion = frame.headers.version;
1591
- // STOMP version 1.2 needs header values to be escaped
1592
- if (this._connectedVersion === _versions__WEBPACK_IMPORTED_MODULE_0__.Versions.V1_2) {
1593
- this._escapeHeaderValues = true;
1594
- }
1595
- this._setupHeartbeat(frame.headers);
1596
- this.onConnect(frame);
1597
- },
1598
- // [MESSAGE Frame](http://stomp.github.com/stomp-specification-1.2.html#MESSAGE)
1599
- MESSAGE: frame => {
1600
- // the callback is registered when the client calls
1601
- // `subscribe()`.
1602
- // If there is no registered subscription for the received message,
1603
- // the default `onUnhandledMessage` callback is used that the client can set.
1604
- // This is useful for subscriptions that are automatically created
1605
- // on the browser side (e.g. [RabbitMQ's temporary
1606
- // queues](http://www.rabbitmq.com/stomp.html)).
1607
- const subscription = frame.headers.subscription;
1608
- const onReceive = this._subscriptions[subscription] || this.onUnhandledMessage;
1609
- // bless the frame to be a Message
1610
- const message = frame;
1611
- const client = this;
1612
- const messageId = this._connectedVersion === _versions__WEBPACK_IMPORTED_MODULE_0__.Versions.V1_2
1613
- ? message.headers.ack
1614
- : message.headers['message-id'];
1615
- // add `ack()` and `nack()` methods directly to the returned frame
1616
- // so that a simple call to `message.ack()` can acknowledge the message.
1617
- message.ack = (headers = {}) => {
1618
- return client.ack(messageId, subscription, headers);
1619
- };
1620
- message.nack = (headers = {}) => {
1621
- return client.nack(messageId, subscription, headers);
1622
- };
1623
- onReceive(message);
1624
- },
1625
- // [RECEIPT Frame](http://stomp.github.com/stomp-specification-1.2.html#RECEIPT)
1626
- RECEIPT: frame => {
1627
- const callback = this._receiptWatchers[frame.headers['receipt-id']];
1628
- if (callback) {
1629
- callback(frame);
1630
- // Server will acknowledge only once, remove the callback
1631
- delete this._receiptWatchers[frame.headers['receipt-id']];
1632
- }
1633
- else {
1634
- this.onUnhandledReceipt(frame);
1635
- }
1636
- },
1637
- // [ERROR Frame](http://stomp.github.com/stomp-specification-1.2.html#ERROR)
1638
- ERROR: frame => {
1639
- this.onStompError(frame);
1640
- },
1641
- };
1642
- // used to index subscribers
1643
- this._counter = 0;
1644
- // subscription callbacks indexed by subscriber's ID
1645
- this._subscriptions = {};
1646
- // receipt-watchers indexed by receipts-ids
1647
- this._receiptWatchers = {};
1648
- this._partialData = '';
1649
- this._escapeHeaderValues = false;
1650
- this._lastServerActivityTS = Date.now();
1651
- this.debug = config.debug;
1652
- this.stompVersions = config.stompVersions;
1653
- this.connectHeaders = config.connectHeaders;
1654
- this.disconnectHeaders = config.disconnectHeaders;
1655
- this.heartbeatIncoming = config.heartbeatIncoming;
1656
- this.heartbeatOutgoing = config.heartbeatOutgoing;
1657
- this.splitLargeFrames = config.splitLargeFrames;
1658
- this.maxWebSocketChunkSize = config.maxWebSocketChunkSize;
1659
- this.forceBinaryWSFrames = config.forceBinaryWSFrames;
1660
- this.logRawCommunication = config.logRawCommunication;
1661
- this.appendMissingNULLonIncoming = config.appendMissingNULLonIncoming;
1662
- this.discardWebsocketOnCommFailure = config.discardWebsocketOnCommFailure;
1663
- this.onConnect = config.onConnect;
1664
- this.onDisconnect = config.onDisconnect;
1665
- this.onStompError = config.onStompError;
1666
- this.onWebSocketClose = config.onWebSocketClose;
1667
- this.onWebSocketError = config.onWebSocketError;
1668
- this.onUnhandledMessage = config.onUnhandledMessage;
1669
- this.onUnhandledReceipt = config.onUnhandledReceipt;
1670
- this.onUnhandledFrame = config.onUnhandledFrame;
1671
- }
1672
- get connectedVersion() {
1673
- return this._connectedVersion;
1674
- }
1675
- get connected() {
1676
- return this._connected;
1677
- }
1678
- start() {
1679
- const parser = new _parser__WEBPACK_IMPORTED_MODULE_1__.Parser(
1680
- // On Frame
1681
- rawFrame => {
1682
- const frame = _frame_impl__WEBPACK_IMPORTED_MODULE_2__.FrameImpl.fromRawFrame(rawFrame, this._escapeHeaderValues);
1683
- // if this.logRawCommunication is set, the rawChunk is logged at this._webSocket.onmessage
1684
- if (!this.logRawCommunication) {
1685
- this.debug(`<<< ${frame}`);
1686
- }
1687
- const serverFrameHandler = this._serverFrameHandlers[frame.command] || this.onUnhandledFrame;
1688
- serverFrameHandler(frame);
1689
- },
1690
- // On Incoming Ping
1691
- () => {
1692
- this.debug('<<< PONG');
1693
- });
1694
- this._webSocket.onmessage = (evt) => {
1695
- this.debug('Received data');
1696
- this._lastServerActivityTS = Date.now();
1697
- if (this.logRawCommunication) {
1698
- const rawChunkAsString = evt.data instanceof ArrayBuffer
1699
- ? new TextDecoder().decode(evt.data)
1700
- : evt.data;
1701
- this.debug(`<<< ${rawChunkAsString}`);
1522
+ /**
1523
+ * Available for backward compatibility, please shift to using {@link Client}.
1524
+ *
1525
+ * **Deprecated**
1526
+ *
1527
+ * Part of `@stomp/stompjs`.
1528
+ *
1529
+ * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)
1530
+ */
1531
+ class CompatClient extends Client {
1532
+ /**
1533
+ * Available for backward compatibility, please shift to using {@link Client}
1534
+ * and [Client#webSocketFactory]{@link Client#webSocketFactory}.
1535
+ *
1536
+ * **Deprecated**
1537
+ *
1538
+ * @internal
1539
+ */
1540
+ constructor(webSocketFactory) {
1541
+ super();
1542
+ /**
1543
+ * It is no op now. No longer needed. Large packets work out of the box.
1544
+ */
1545
+ this.maxWebSocketFrameSize = 16 * 1024;
1546
+ this._heartbeatInfo = new HeartbeatInfo(this);
1547
+ this.reconnect_delay = 0;
1548
+ this.webSocketFactory = webSocketFactory;
1549
+ // Default from previous version
1550
+ this.debug = (...message) => {
1551
+ console.log(...message);
1552
+ };
1553
+ }
1554
+ _parseConnect(...args) {
1555
+ let closeEventCallback;
1556
+ let connectCallback;
1557
+ let errorCallback;
1558
+ let headers = {};
1559
+ if (args.length < 2) {
1560
+ throw new Error('Connect requires at least 2 arguments');
1702
1561
  }
1703
- parser.parseChunk(evt.data, this.appendMissingNULLonIncoming);
1704
- };
1705
- this._webSocket.onclose = (closeEvent) => {
1706
- this.debug(`Connection closed to ${this._client.brokerURL}`);
1707
- this._cleanUp();
1708
- this.onWebSocketClose(closeEvent);
1709
- };
1710
- this._webSocket.onerror = (errorEvent) => {
1711
- this.onWebSocketError(errorEvent);
1712
- };
1713
- this._webSocket.onopen = () => {
1714
- // Clone before updating
1715
- const connectHeaders = Object.assign({}, this.connectHeaders);
1716
- this.debug('Web Socket Opened...');
1717
- connectHeaders['accept-version'] = this.stompVersions.supportedVersions();
1718
- connectHeaders['heart-beat'] = [
1719
- this.heartbeatOutgoing,
1720
- this.heartbeatIncoming,
1721
- ].join(',');
1722
- this._transmit({ command: 'CONNECT', headers: connectHeaders });
1723
- };
1724
- }
1725
- _setupHeartbeat(headers) {
1726
- if (headers.version !== _versions__WEBPACK_IMPORTED_MODULE_0__.Versions.V1_1 &&
1727
- headers.version !== _versions__WEBPACK_IMPORTED_MODULE_0__.Versions.V1_2) {
1728
- return;
1729
- }
1730
- // It is valid for the server to not send this header
1731
- // https://stomp.github.io/stomp-specification-1.2.html#Heart-beating
1732
- if (!headers['heart-beat']) {
1733
- return;
1734
- }
1735
- // heart-beat header received from the server looks like:
1736
- //
1737
- // heart-beat: sx, sy
1738
- const [serverOutgoing, serverIncoming] = headers['heart-beat']
1739
- .split(',')
1740
- .map((v) => parseInt(v, 10));
1741
- if (this.heartbeatOutgoing !== 0 && serverIncoming !== 0) {
1742
- const ttl = Math.max(this.heartbeatOutgoing, serverIncoming);
1743
- this.debug(`send PING every ${ttl}ms`);
1744
- this._pinger = setInterval(() => {
1745
- if (this._webSocket.readyState === _types__WEBPACK_IMPORTED_MODULE_3__.StompSocketState.OPEN) {
1746
- this._webSocket.send(_byte__WEBPACK_IMPORTED_MODULE_4__.BYTE.LF);
1747
- this.debug('>>> PING');
1748
- }
1749
- }, ttl);
1750
- }
1751
- if (this.heartbeatIncoming !== 0 && serverOutgoing !== 0) {
1752
- const ttl = Math.max(this.heartbeatIncoming, serverOutgoing);
1753
- this.debug(`check PONG every ${ttl}ms`);
1754
- this._ponger = setInterval(() => {
1755
- const delta = Date.now() - this._lastServerActivityTS;
1756
- // We wait twice the TTL to be flexible on window's setInterval calls
1757
- if (delta > ttl * 2) {
1758
- this.debug(`did not receive server activity for the last ${delta}ms`);
1759
- this._closeOrDiscardWebsocket();
1562
+ if (typeof args[1] === 'function') {
1563
+ [headers, connectCallback, errorCallback, closeEventCallback] = args;
1564
+ }
1565
+ else {
1566
+ switch (args.length) {
1567
+ case 6:
1568
+ [
1569
+ headers.login,
1570
+ headers.passcode,
1571
+ connectCallback,
1572
+ errorCallback,
1573
+ closeEventCallback,
1574
+ headers.host,
1575
+ ] = args;
1576
+ break;
1577
+ default:
1578
+ [
1579
+ headers.login,
1580
+ headers.passcode,
1581
+ connectCallback,
1582
+ errorCallback,
1583
+ closeEventCallback,
1584
+ ] = args;
1760
1585
  }
1761
- }, ttl);
1762
- }
1763
- }
1764
- _closeOrDiscardWebsocket() {
1765
- if (this.discardWebsocketOnCommFailure) {
1766
- this.debug('Discarding websocket, the underlying socket may linger for a while');
1767
- this.discardWebsocket();
1768
- }
1769
- else {
1770
- this.debug('Issuing close on the websocket');
1771
- this._closeWebsocket();
1772
- }
1773
- }
1774
- forceDisconnect() {
1775
- if (this._webSocket) {
1776
- if (this._webSocket.readyState === _types__WEBPACK_IMPORTED_MODULE_3__.StompSocketState.CONNECTING ||
1777
- this._webSocket.readyState === _types__WEBPACK_IMPORTED_MODULE_3__.StompSocketState.OPEN) {
1778
- this._closeOrDiscardWebsocket();
1779
1586
  }
1587
+ return [headers, connectCallback, errorCallback, closeEventCallback];
1780
1588
  }
1781
- }
1782
- _closeWebsocket() {
1783
- this._webSocket.onmessage = () => { }; // ignore messages
1784
- this._webSocket.close();
1785
- }
1786
- discardWebsocket() {
1787
- if (typeof this._webSocket.terminate !== 'function') {
1788
- (0,_augment_websocket__WEBPACK_IMPORTED_MODULE_5__.augmentWebsocket)(this._webSocket, (msg) => this.debug(msg));
1789
- }
1790
- // @ts-ignore - this method will be there at this stage
1791
- this._webSocket.terminate();
1792
- }
1793
- _transmit(params) {
1794
- const { command, headers, body, binaryBody, skipContentLengthHeader } = params;
1795
- const frame = new _frame_impl__WEBPACK_IMPORTED_MODULE_2__.FrameImpl({
1796
- command,
1797
- headers,
1798
- body,
1799
- binaryBody,
1800
- escapeHeaderValues: this._escapeHeaderValues,
1801
- skipContentLengthHeader,
1802
- });
1803
- let rawChunk = frame.serialize();
1804
- if (this.logRawCommunication) {
1805
- this.debug(`>>> ${rawChunk}`);
1806
- }
1807
- else {
1808
- this.debug(`>>> ${frame}`);
1809
- }
1810
- if (this.forceBinaryWSFrames && typeof rawChunk === 'string') {
1811
- rawChunk = new TextEncoder().encode(rawChunk);
1812
- }
1813
- if (typeof rawChunk !== 'string' || !this.splitLargeFrames) {
1814
- this._webSocket.send(rawChunk);
1815
- }
1816
- else {
1817
- let out = rawChunk;
1818
- while (out.length > 0) {
1819
- const chunk = out.substring(0, this.maxWebSocketChunkSize);
1820
- out = out.substring(this.maxWebSocketChunkSize);
1821
- this._webSocket.send(chunk);
1822
- this.debug(`chunk sent = ${chunk.length}, remaining = ${out.length}`);
1589
+ /**
1590
+ * Available for backward compatibility, please shift to using [Client#activate]{@link Client#activate}.
1591
+ *
1592
+ * **Deprecated**
1593
+ *
1594
+ * The `connect` method accepts different number of arguments and types. See the Overloads list. Use the
1595
+ * version with headers to pass your broker specific options.
1596
+ *
1597
+ * overloads:
1598
+ * - connect(headers, connectCallback)
1599
+ * - connect(headers, connectCallback, errorCallback)
1600
+ * - connect(login, passcode, connectCallback)
1601
+ * - connect(login, passcode, connectCallback, errorCallback)
1602
+ * - connect(login, passcode, connectCallback, errorCallback, closeEventCallback)
1603
+ * - connect(login, passcode, connectCallback, errorCallback, closeEventCallback, host)
1604
+ *
1605
+ * params:
1606
+ * - headers, see [Client#connectHeaders]{@link Client#connectHeaders}
1607
+ * - connectCallback, see [Client#onConnect]{@link Client#onConnect}
1608
+ * - errorCallback, see [Client#onStompError]{@link Client#onStompError}
1609
+ * - closeEventCallback, see [Client#onWebSocketClose]{@link Client#onWebSocketClose}
1610
+ * - login [String], see [Client#connectHeaders](../classes/Client.html#connectHeaders)
1611
+ * - passcode [String], [Client#connectHeaders](../classes/Client.html#connectHeaders)
1612
+ * - host [String], see [Client#connectHeaders](../classes/Client.html#connectHeaders)
1613
+ *
1614
+ * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)
1615
+ */
1616
+ connect(...args) {
1617
+ const out = this._parseConnect(...args);
1618
+ if (out[0]) {
1619
+ this.connectHeaders = out[0];
1823
1620
  }
1824
- }
1825
- }
1826
- dispose() {
1827
- if (this.connected) {
1828
- try {
1829
- // clone before updating
1830
- const disconnectHeaders = Object.assign({}, this.disconnectHeaders);
1831
- if (!disconnectHeaders.receipt) {
1832
- disconnectHeaders.receipt = `close-${this._counter++}`;
1833
- }
1834
- this.watchForReceipt(disconnectHeaders.receipt, frame => {
1835
- this._closeWebsocket();
1836
- this._cleanUp();
1837
- this.onDisconnect(frame);
1838
- });
1839
- this._transmit({ command: 'DISCONNECT', headers: disconnectHeaders });
1621
+ if (out[1]) {
1622
+ this.onConnect = out[1];
1623
+ }
1624
+ if (out[2]) {
1625
+ this.onStompError = out[2];
1840
1626
  }
1841
- catch (error) {
1842
- this.debug(`Ignoring error during disconnect ${error}`);
1627
+ if (out[3]) {
1628
+ this.onWebSocketClose = out[3];
1843
1629
  }
1630
+ super.activate();
1844
1631
  }
1845
- else {
1846
- if (this._webSocket.readyState === _types__WEBPACK_IMPORTED_MODULE_3__.StompSocketState.CONNECTING ||
1847
- this._webSocket.readyState === _types__WEBPACK_IMPORTED_MODULE_3__.StompSocketState.OPEN) {
1848
- this._closeWebsocket();
1632
+ /**
1633
+ * Available for backward compatibility, please shift to using [Client#deactivate]{@link Client#deactivate}.
1634
+ *
1635
+ * **Deprecated**
1636
+ *
1637
+ * See:
1638
+ * [Client#onDisconnect]{@link Client#onDisconnect}, and
1639
+ * [Client#disconnectHeaders]{@link Client#disconnectHeaders}
1640
+ *
1641
+ * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)
1642
+ */
1643
+ disconnect(disconnectCallback, headers = {}) {
1644
+ if (disconnectCallback) {
1645
+ this.onDisconnect = disconnectCallback;
1849
1646
  }
1647
+ this.disconnectHeaders = headers;
1648
+ super.deactivate();
1850
1649
  }
1851
- }
1852
- _cleanUp() {
1853
- this._connected = false;
1854
- if (this._pinger) {
1855
- clearInterval(this._pinger);
1856
- this._pinger = undefined;
1650
+ /**
1651
+ * Available for backward compatibility, use [Client#publish]{@link Client#publish}.
1652
+ *
1653
+ * Send a message to a named destination. Refer to your STOMP broker documentation for types
1654
+ * and naming of destinations. The headers will, typically, be available to the subscriber.
1655
+ * However, there may be special purpose headers corresponding to your STOMP broker.
1656
+ *
1657
+ * **Deprecated**, use [Client#publish]{@link Client#publish}
1658
+ *
1659
+ * Note: Body must be String. You will need to covert the payload to string in case it is not string (e.g. JSON)
1660
+ *
1661
+ * ```javascript
1662
+ * client.send("/queue/test", {priority: 9}, "Hello, STOMP");
1663
+ *
1664
+ * // If you want to send a message with a body, you must also pass the headers argument.
1665
+ * client.send("/queue/test", {}, "Hello, STOMP");
1666
+ * ```
1667
+ *
1668
+ * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)
1669
+ */
1670
+ send(destination, headers = {}, body = '') {
1671
+ headers = Object.assign({}, headers);
1672
+ const skipContentLengthHeader = headers['content-length'] === false;
1673
+ if (skipContentLengthHeader) {
1674
+ delete headers['content-length'];
1675
+ }
1676
+ this.publish({
1677
+ destination,
1678
+ headers: headers,
1679
+ body,
1680
+ skipContentLengthHeader,
1681
+ });
1857
1682
  }
1858
- if (this._ponger) {
1859
- clearInterval(this._ponger);
1860
- this._ponger = undefined;
1683
+ /**
1684
+ * Available for backward compatibility, renamed to [Client#reconnectDelay]{@link Client#reconnectDelay}.
1685
+ *
1686
+ * **Deprecated**
1687
+ */
1688
+ set reconnect_delay(value) {
1689
+ this.reconnectDelay = value;
1861
1690
  }
1862
- }
1863
- publish(params) {
1864
- const { destination, headers, body, binaryBody, skipContentLengthHeader } = params;
1865
- const hdrs = Object.assign({ destination }, headers);
1866
- this._transmit({
1867
- command: 'SEND',
1868
- headers: hdrs,
1869
- body,
1870
- binaryBody,
1871
- skipContentLengthHeader,
1872
- });
1873
- }
1874
- watchForReceipt(receiptId, callback) {
1875
- this._receiptWatchers[receiptId] = callback;
1876
- }
1877
- subscribe(destination, callback, headers = {}) {
1878
- headers = Object.assign({}, headers);
1879
- if (!headers.id) {
1880
- headers.id = `sub-${this._counter++}`;
1881
- }
1882
- headers.destination = destination;
1883
- this._subscriptions[headers.id] = callback;
1884
- this._transmit({ command: 'SUBSCRIBE', headers });
1885
- const client = this;
1886
- return {
1887
- id: headers.id,
1888
- unsubscribe(hdrs) {
1889
- return client.unsubscribe(headers.id, hdrs);
1890
- },
1891
- };
1892
- }
1893
- unsubscribe(id, headers = {}) {
1894
- headers = Object.assign({}, headers);
1895
- delete this._subscriptions[id];
1896
- headers.id = id;
1897
- this._transmit({ command: 'UNSUBSCRIBE', headers });
1898
- }
1899
- begin(transactionId) {
1900
- const txId = transactionId || `tx-${this._counter++}`;
1901
- this._transmit({
1902
- command: 'BEGIN',
1903
- headers: {
1904
- transaction: txId,
1905
- },
1906
- });
1907
- const client = this;
1908
- return {
1909
- id: txId,
1910
- commit() {
1911
- client.commit(txId);
1912
- },
1913
- abort() {
1914
- client.abort(txId);
1915
- },
1916
- };
1917
- }
1918
- commit(transactionId) {
1919
- this._transmit({
1920
- command: 'COMMIT',
1921
- headers: {
1922
- transaction: transactionId,
1923
- },
1924
- });
1925
- }
1926
- abort(transactionId) {
1927
- this._transmit({
1928
- command: 'ABORT',
1929
- headers: {
1930
- transaction: transactionId,
1931
- },
1932
- });
1933
- }
1934
- ack(messageId, subscriptionId, headers = {}) {
1935
- headers = Object.assign({}, headers);
1936
- if (this._connectedVersion === _versions__WEBPACK_IMPORTED_MODULE_0__.Versions.V1_2) {
1937
- headers.id = messageId;
1691
+ /**
1692
+ * Available for backward compatibility, renamed to [Client#webSocket]{@link Client#webSocket}.
1693
+ *
1694
+ * **Deprecated**
1695
+ */
1696
+ get ws() {
1697
+ return this.webSocket;
1938
1698
  }
1939
- else {
1940
- headers['message-id'] = messageId;
1699
+ /**
1700
+ * Available for backward compatibility, renamed to [Client#connectedVersion]{@link Client#connectedVersion}.
1701
+ *
1702
+ * **Deprecated**
1703
+ */
1704
+ get version() {
1705
+ return this.connectedVersion;
1941
1706
  }
1942
- headers.subscription = subscriptionId;
1943
- this._transmit({ command: 'ACK', headers });
1944
- }
1945
- nack(messageId, subscriptionId, headers = {}) {
1946
- headers = Object.assign({}, headers);
1947
- if (this._connectedVersion === _versions__WEBPACK_IMPORTED_MODULE_0__.Versions.V1_2) {
1948
- headers.id = messageId;
1707
+ /**
1708
+ * Available for backward compatibility, renamed to [Client#onUnhandledMessage]{@link Client#onUnhandledMessage}.
1709
+ *
1710
+ * **Deprecated**
1711
+ */
1712
+ get onreceive() {
1713
+ return this.onUnhandledMessage;
1714
+ }
1715
+ /**
1716
+ * Available for backward compatibility, renamed to [Client#onUnhandledMessage]{@link Client#onUnhandledMessage}.
1717
+ *
1718
+ * **Deprecated**
1719
+ */
1720
+ set onreceive(value) {
1721
+ this.onUnhandledMessage = value;
1722
+ }
1723
+ /**
1724
+ * Available for backward compatibility, renamed to [Client#onUnhandledReceipt]{@link Client#onUnhandledReceipt}.
1725
+ * Prefer using [Client#watchForReceipt]{@link Client#watchForReceipt}.
1726
+ *
1727
+ * **Deprecated**
1728
+ */
1729
+ get onreceipt() {
1730
+ return this.onUnhandledReceipt;
1731
+ }
1732
+ /**
1733
+ * Available for backward compatibility, renamed to [Client#onUnhandledReceipt]{@link Client#onUnhandledReceipt}.
1734
+ *
1735
+ * **Deprecated**
1736
+ */
1737
+ set onreceipt(value) {
1738
+ this.onUnhandledReceipt = value;
1739
+ }
1740
+ /**
1741
+ * Available for backward compatibility, renamed to [Client#heartbeatIncoming]{@link Client#heartbeatIncoming}
1742
+ * [Client#heartbeatOutgoing]{@link Client#heartbeatOutgoing}.
1743
+ *
1744
+ * **Deprecated**
1745
+ */
1746
+ get heartbeat() {
1747
+ return this._heartbeatInfo;
1949
1748
  }
1950
- else {
1951
- headers['message-id'] = messageId;
1749
+ /**
1750
+ * Available for backward compatibility, renamed to [Client#heartbeatIncoming]{@link Client#heartbeatIncoming}
1751
+ * [Client#heartbeatOutgoing]{@link Client#heartbeatOutgoing}.
1752
+ *
1753
+ * **Deprecated**
1754
+ */
1755
+ set heartbeat(value) {
1756
+ this.heartbeatIncoming = value.incoming;
1757
+ this.heartbeatOutgoing = value.outgoing;
1952
1758
  }
1953
- headers.subscription = subscriptionId;
1954
- return this._transmit({ command: 'NACK', headers });
1955
1759
  }
1956
- }
1957
-
1958
-
1959
- /***/ }),
1960
-
1961
- /***/ "./src/stomp-headers.ts":
1962
- /*!******************************!*\
1963
- !*** ./src/stomp-headers.ts ***!
1964
- \******************************/
1965
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1966
1760
 
1967
- __webpack_require__.r(__webpack_exports__);
1968
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1969
- /* harmony export */ "StompHeaders": () => (/* binding */ StompHeaders)
1970
- /* harmony export */ });
1971
- /**
1972
- * STOMP headers. Many functions calls will accept headers as parameters.
1973
- * The headers sent by Broker will be available as [IFrame#headers]{@link IFrame#headers}.
1974
- *
1975
- * `key` and `value` must be valid strings.
1976
- * In addition, `key` must not contain `CR`, `LF`, or `:`.
1977
- *
1978
- * Part of `@stomp/stompjs`.
1979
- */
1980
- class StompHeaders {
1981
- }
1982
-
1983
-
1984
- /***/ }),
1985
-
1986
- /***/ "./src/types.ts":
1987
- /*!**********************!*\
1988
- !*** ./src/types.ts ***!
1989
- \**********************/
1990
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
1991
-
1992
- __webpack_require__.r(__webpack_exports__);
1993
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
1994
- /* harmony export */ "ActivationState": () => (/* binding */ ActivationState),
1995
- /* harmony export */ "StompSocketState": () => (/* binding */ StompSocketState)
1996
- /* harmony export */ });
1997
- /**
1998
- * Possible states for the IStompSocket
1999
- */
2000
- var StompSocketState;
2001
- (function (StompSocketState) {
2002
- StompSocketState[StompSocketState["CONNECTING"] = 0] = "CONNECTING";
2003
- StompSocketState[StompSocketState["OPEN"] = 1] = "OPEN";
2004
- StompSocketState[StompSocketState["CLOSING"] = 2] = "CLOSING";
2005
- StompSocketState[StompSocketState["CLOSED"] = 3] = "CLOSED";
2006
- })(StompSocketState || (StompSocketState = {}));
2007
- /**
2008
- * Possible activation state
2009
- */
2010
- var ActivationState;
2011
- (function (ActivationState) {
2012
- ActivationState[ActivationState["ACTIVE"] = 0] = "ACTIVE";
2013
- ActivationState[ActivationState["DEACTIVATING"] = 1] = "DEACTIVATING";
2014
- ActivationState[ActivationState["INACTIVE"] = 2] = "INACTIVE";
2015
- })(ActivationState || (ActivationState = {}));
2016
-
2017
-
2018
- /***/ }),
2019
-
2020
- /***/ "./src/versions.ts":
2021
- /*!*************************!*\
2022
- !*** ./src/versions.ts ***!
2023
- \*************************/
2024
- /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
2025
-
2026
- __webpack_require__.r(__webpack_exports__);
2027
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
2028
- /* harmony export */ "Versions": () => (/* binding */ Versions)
2029
- /* harmony export */ });
2030
- /**
2031
- * Supported STOMP versions
2032
- *
2033
- * Part of `@stomp/stompjs`.
2034
- */
2035
- class Versions {
2036
1761
  /**
2037
- * Takes an array of string of versions, typical elements '1.0', '1.1', or '1.2'
1762
+ * STOMP Class, acts like a factory to create {@link Client}.
2038
1763
  *
2039
- * You will an instance if this class if you want to override supported versions to be declared during
2040
- * STOMP handshake.
2041
- */
2042
- constructor(versions) {
2043
- this.versions = versions;
2044
- }
2045
- /**
2046
- * Used as part of CONNECT STOMP Frame
1764
+ * Part of `@stomp/stompjs`.
1765
+ *
1766
+ * **Deprecated**
1767
+ *
1768
+ * It will be removed in next major version. Please switch to {@link Client}.
2047
1769
  */
2048
- supportedVersions() {
2049
- return this.versions.join(',');
1770
+ class Stomp {
1771
+ /**
1772
+ * This method creates a WebSocket client that is connected to
1773
+ * the STOMP server located at the url.
1774
+ *
1775
+ * ```javascript
1776
+ * var url = "ws://localhost:61614/stomp";
1777
+ * var client = Stomp.client(url);
1778
+ * ```
1779
+ *
1780
+ * **Deprecated**
1781
+ *
1782
+ * It will be removed in next major version. Please switch to {@link Client}
1783
+ * using [Client#brokerURL]{@link Client#brokerURL}.
1784
+ */
1785
+ static client(url, protocols) {
1786
+ // This is a hack to allow another implementation than the standard
1787
+ // HTML5 WebSocket class.
1788
+ //
1789
+ // It is possible to use another class by calling
1790
+ //
1791
+ // Stomp.WebSocketClass = MozWebSocket
1792
+ //
1793
+ // *prior* to call `Stomp.client()`.
1794
+ //
1795
+ // This hack is deprecated and `Stomp.over()` method should be used
1796
+ // instead.
1797
+ // See remarks on the function Stomp.over
1798
+ if (protocols == null) {
1799
+ protocols = Versions.default.protocolVersions();
1800
+ }
1801
+ const wsFn = () => {
1802
+ const klass = Stomp.WebSocketClass || WebSocket;
1803
+ return new klass(url, protocols);
1804
+ };
1805
+ return new CompatClient(wsFn);
1806
+ }
1807
+ /**
1808
+ * This method is an alternative to [Stomp#client]{@link Stomp#client} to let the user
1809
+ * specify the WebSocket to use (either a standard HTML5 WebSocket or
1810
+ * a similar object).
1811
+ *
1812
+ * In order to support reconnection, the function Client._connect should be callable more than once.
1813
+ * While reconnecting
1814
+ * a new instance of underlying transport (TCP Socket, WebSocket or SockJS) will be needed. So, this function
1815
+ * alternatively allows passing a function that should return a new instance of the underlying socket.
1816
+ *
1817
+ * ```javascript
1818
+ * var client = Stomp.over(function(){
1819
+ * return new WebSocket('ws://localhost:15674/ws')
1820
+ * });
1821
+ * ```
1822
+ *
1823
+ * **Deprecated**
1824
+ *
1825
+ * It will be removed in next major version. Please switch to {@link Client}
1826
+ * using [Client#webSocketFactory]{@link Client#webSocketFactory}.
1827
+ */
1828
+ static over(ws) {
1829
+ let wsFn;
1830
+ if (typeof ws === 'function') {
1831
+ wsFn = ws;
1832
+ }
1833
+ else {
1834
+ console.warn('Stomp.over did not receive a factory, auto reconnect will not work. ' +
1835
+ 'Please see https://stomp-js.github.io/api-docs/latest/classes/Stomp.html#over');
1836
+ wsFn = () => ws;
1837
+ }
1838
+ return new CompatClient(wsFn);
1839
+ }
2050
1840
  }
2051
1841
  /**
2052
- * Used while creating a WebSocket
1842
+ * In case you need to use a non standard class for WebSocket.
1843
+ *
1844
+ * For example when using within NodeJS environment:
1845
+ *
1846
+ * ```javascript
1847
+ * StompJs = require('../../esm5/');
1848
+ * Stomp = StompJs.Stomp;
1849
+ * Stomp.WebSocketClass = require('websocket').w3cwebsocket;
1850
+ * ```
1851
+ *
1852
+ * **Deprecated**
1853
+ *
1854
+ *
1855
+ * It will be removed in next major version. Please switch to {@link Client}
1856
+ * using [Client#webSocketFactory]{@link Client#webSocketFactory}.
2053
1857
  */
2054
- protocolVersions() {
2055
- return this.versions.map(x => `v${x.replace('.', '')}.stomp`);
2056
- }
2057
- }
2058
- /**
2059
- * Indicates protocol version 1.0
2060
- */
2061
- Versions.V1_0 = '1.0';
2062
- /**
2063
- * Indicates protocol version 1.1
2064
- */
2065
- Versions.V1_1 = '1.1';
2066
- /**
2067
- * Indicates protocol version 1.2
2068
- */
2069
- Versions.V1_2 = '1.2';
2070
- /**
2071
- * @internal
2072
- */
2073
- Versions.default = new Versions([
2074
- Versions.V1_2,
2075
- Versions.V1_1,
2076
- Versions.V1_0,
2077
- ]);
2078
-
2079
-
2080
- /***/ })
2081
-
2082
- /******/ });
2083
- /************************************************************************/
2084
- /******/ // The module cache
2085
- /******/ var __webpack_module_cache__ = {};
2086
- /******/
2087
- /******/ // The require function
2088
- /******/ function __webpack_require__(moduleId) {
2089
- /******/ // Check if module is in cache
2090
- /******/ var cachedModule = __webpack_module_cache__[moduleId];
2091
- /******/ if (cachedModule !== undefined) {
2092
- /******/ return cachedModule.exports;
2093
- /******/ }
2094
- /******/ // Create a new module (and put it into the cache)
2095
- /******/ var module = __webpack_module_cache__[moduleId] = {
2096
- /******/ // no module.id needed
2097
- /******/ // no module.loaded needed
2098
- /******/ exports: {}
2099
- /******/ };
2100
- /******/
2101
- /******/ // Execute the module function
2102
- /******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
2103
- /******/
2104
- /******/ // Return the exports of the module
2105
- /******/ return module.exports;
2106
- /******/ }
2107
- /******/
2108
- /************************************************************************/
2109
- /******/ /* webpack/runtime/define property getters */
2110
- /******/ (() => {
2111
- /******/ // define getter functions for harmony exports
2112
- /******/ __webpack_require__.d = (exports, definition) => {
2113
- /******/ for(var key in definition) {
2114
- /******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {
2115
- /******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] });
2116
- /******/ }
2117
- /******/ }
2118
- /******/ };
2119
- /******/ })();
2120
- /******/
2121
- /******/ /* webpack/runtime/hasOwnProperty shorthand */
2122
- /******/ (() => {
2123
- /******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))
2124
- /******/ })();
2125
- /******/
2126
- /******/ /* webpack/runtime/make namespace object */
2127
- /******/ (() => {
2128
- /******/ // define __esModule on exports
2129
- /******/ __webpack_require__.r = (exports) => {
2130
- /******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
2131
- /******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2132
- /******/ }
2133
- /******/ Object.defineProperty(exports, '__esModule', { value: true });
2134
- /******/ };
2135
- /******/ })();
2136
- /******/
2137
- /************************************************************************/
2138
- var __webpack_exports__ = {};
2139
- // This entry need to be wrapped in an IIFE because it need to be isolated against other modules in the chunk.
2140
- (() => {
2141
- /*!**********************!*\
2142
- !*** ./src/index.ts ***!
2143
- \**********************/
2144
- __webpack_require__.r(__webpack_exports__);
2145
- /* harmony export */ __webpack_require__.d(__webpack_exports__, {
2146
- /* harmony export */ "ActivationState": () => (/* reexport safe */ _types__WEBPACK_IMPORTED_MODULE_5__.ActivationState),
2147
- /* harmony export */ "Client": () => (/* reexport safe */ _client__WEBPACK_IMPORTED_MODULE_0__.Client),
2148
- /* harmony export */ "CompatClient": () => (/* reexport safe */ _compatibility_compat_client__WEBPACK_IMPORTED_MODULE_7__.CompatClient),
2149
- /* harmony export */ "FrameImpl": () => (/* reexport safe */ _frame_impl__WEBPACK_IMPORTED_MODULE_1__.FrameImpl),
2150
- /* harmony export */ "Parser": () => (/* reexport safe */ _parser__WEBPACK_IMPORTED_MODULE_2__.Parser),
2151
- /* harmony export */ "Stomp": () => (/* reexport safe */ _compatibility_stomp__WEBPACK_IMPORTED_MODULE_8__.Stomp),
2152
- /* harmony export */ "StompConfig": () => (/* reexport safe */ _stomp_config__WEBPACK_IMPORTED_MODULE_3__.StompConfig),
2153
- /* harmony export */ "StompHeaders": () => (/* reexport safe */ _stomp_headers__WEBPACK_IMPORTED_MODULE_4__.StompHeaders),
2154
- /* harmony export */ "StompSocketState": () => (/* reexport safe */ _types__WEBPACK_IMPORTED_MODULE_5__.StompSocketState),
2155
- /* harmony export */ "Versions": () => (/* reexport safe */ _versions__WEBPACK_IMPORTED_MODULE_6__.Versions)
2156
- /* harmony export */ });
2157
- /* harmony import */ var _client__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./client */ "./src/client.ts");
2158
- /* harmony import */ var _frame_impl__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./frame-impl */ "./src/frame-impl.ts");
2159
- /* harmony import */ var _parser__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./parser */ "./src/parser.ts");
2160
- /* harmony import */ var _stomp_config__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./stomp-config */ "./src/stomp-config.ts");
2161
- /* harmony import */ var _stomp_headers__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./stomp-headers */ "./src/stomp-headers.ts");
2162
- /* harmony import */ var _types__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./types */ "./src/types.ts");
2163
- /* harmony import */ var _versions__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./versions */ "./src/versions.ts");
2164
- /* harmony import */ var _compatibility_compat_client__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./compatibility/compat-client */ "./src/compatibility/compat-client.ts");
2165
- /* harmony import */ var _compatibility_stomp__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./compatibility/stomp */ "./src/compatibility/stomp.ts");
2166
-
2167
-
2168
-
2169
-
2170
-
2171
-
2172
-
2173
-
2174
-
2175
-
2176
-
2177
- // Compatibility code
2178
-
2179
-
2180
-
2181
- })();
2182
-
2183
- /******/ return __webpack_exports__;
2184
- /******/ })()
2185
- ;
2186
- });
2187
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"stomp.umd.js","mappings":"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","sources":["webpack://StompJs/webpack/universalModuleDefinition","webpack://StompJs/./src/augment-websocket.ts","webpack://StompJs/./src/byte.ts","webpack://StompJs/./src/client.ts","webpack://StompJs/./src/compatibility/compat-client.ts","webpack://StompJs/./src/compatibility/heartbeat-info.ts","webpack://StompJs/./src/compatibility/stomp.ts","webpack://StompJs/./src/frame-impl.ts","webpack://StompJs/./src/parser.ts","webpack://StompJs/./src/stomp-config.ts","webpack://StompJs/./src/stomp-handler.ts","webpack://StompJs/./src/stomp-headers.ts","webpack://StompJs/./src/types.ts","webpack://StompJs/./src/versions.ts","webpack://StompJs/webpack/bootstrap","webpack://StompJs/webpack/runtime/define property getters","webpack://StompJs/webpack/runtime/hasOwnProperty shorthand","webpack://StompJs/webpack/runtime/make namespace object","webpack://StompJs/./src/index.ts"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine(\"StompJs\", [], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"StompJs\"] = factory();\n\telse\n\t\troot[\"StompJs\"] = factory();\n})(typeof self !== 'undefined' ? self : this, () => {\nreturn ","import { IStompSocket } from './types';\n\n/**\n * @internal\n */\nexport function augmentWebsocket(\n  webSocket: IStompSocket,\n  debug: (msg: string) => void\n) {\n  webSocket.terminate = function () {\n    const noOp = () => {};\n\n    // set all callbacks to no op\n    this.onerror = noOp;\n    this.onmessage = noOp;\n    this.onopen = noOp;\n\n    const ts = new Date();\n    const id = Math.random().toString().substring(2, 8); // A simulated id\n\n    const origOnClose = this.onclose;\n\n    // Track delay in actual closure of the socket\n    this.onclose = closeEvent => {\n      const delay = new Date().getTime() - ts.getTime();\n      debug(\n        `Discarded socket (#${id})  closed after ${delay}ms, with code/reason: ${closeEvent.code}/${closeEvent.reason}`\n      );\n    };\n\n    this.close();\n\n    origOnClose?.call(webSocket, {\n      code: 4001,\n      reason: `Quick discarding socket (#${id}) without waiting for the shutdown sequence.`,\n      wasClean: false,\n    });\n  };\n}\n","/**\n * Some byte values, used as per STOMP specifications.\n *\n * Part of `@stomp/stompjs`.\n *\n * @internal\n */\nexport const BYTE = {\n  // LINEFEED byte (octet 10)\n  LF: '\\x0A',\n  // NULL byte (octet 0)\n  NULL: '\\x00',\n};\n","import { ITransaction } from './i-transaction';\nimport { StompConfig } from './stomp-config';\nimport { StompHandler } from './stomp-handler';\nimport { StompHeaders } from './stomp-headers';\nimport { StompSubscription } from './stomp-subscription';\nimport {\n  ActivationState,\n  closeEventCallbackType,\n  debugFnType,\n  frameCallbackType,\n  IPublishParams,\n  IStompSocket,\n  messageCallbackType,\n  StompSocketState,\n  wsErrorCallbackType,\n} from './types';\nimport { Versions } from './versions';\n\n/**\n * @internal\n */\ndeclare const WebSocket: {\n  prototype: IStompSocket;\n  new (url: string, protocols?: string | string[]): IStompSocket;\n};\n\n/**\n * STOMP Client Class.\n *\n * Part of `@stomp/stompjs`.\n */\nexport class Client {\n  /**\n   * The URL for the STOMP broker to connect to.\n   * Typically like `\"ws://broker.329broker.com:15674/ws\"` or `\"wss://broker.329broker.com:15674/ws\"`.\n   *\n   * Only one of this or [Client#webSocketFactory]{@link Client#webSocketFactory} need to be set.\n   * If both are set, [Client#webSocketFactory]{@link Client#webSocketFactory} will be used.\n   *\n   * If your environment does not support WebSockets natively, please refer to\n   * [Polyfills]{@link https://stomp-js.github.io/guide/stompjs/rx-stomp/ng2-stompjs/pollyfils-for-stompjs-v5.html}.\n   */\n  public brokerURL: string | undefined;\n\n  /**\n   * STOMP versions to attempt during STOMP handshake. By default versions `1.0`, `1.1`, and `1.2` are attempted.\n   *\n   * Example:\n   * ```javascript\n   *        // Try only versions 1.0 and 1.1\n   *        client.stompVersions = new Versions(['1.0', '1.1'])\n   * ```\n   */\n  public stompVersions = Versions.default;\n\n  /**\n   * This function should return a WebSocket or a similar (e.g. SockJS) object.\n   * If your environment does not support WebSockets natively, please refer to\n   * [Polyfills]{@link https://stomp-js.github.io/guide/stompjs/rx-stomp/ng2-stompjs/pollyfils-for-stompjs-v5.html}.\n   * If your STOMP Broker supports WebSockets, prefer setting [Client#brokerURL]{@link Client#brokerURL}.\n   *\n   * If both this and [Client#brokerURL]{@link Client#brokerURL} are set, this will be used.\n   *\n   * Example:\n   * ```javascript\n   *        // use a WebSocket\n   *        client.webSocketFactory= function () {\n   *          return new WebSocket(\"wss://broker.329broker.com:15674/ws\");\n   *        };\n   *\n   *        // Typical usage with SockJS\n   *        client.webSocketFactory= function () {\n   *          return new SockJS(\"http://broker.329broker.com/stomp\");\n   *        };\n   * ```\n   */\n  public webSocketFactory: (() => IStompSocket) | undefined;\n\n  /**\n   * Will retry if Stomp connection is not established in specified milliseconds.\n   * Default 0, which implies wait for ever.\n   */\n  public connectionTimeout: number = 0;\n\n  // As per https://stackoverflow.com/questions/45802988/typescript-use-correct-version-of-settimeout-node-vs-window/56239226#56239226\n  private _connectionWatcher: ReturnType<typeof setTimeout> | undefined; // Timer\n\n  /**\n   *  automatically reconnect with delay in milliseconds, set to 0 to disable.\n   */\n  public reconnectDelay: number = 5000;\n\n  /**\n   * Incoming heartbeat interval in milliseconds. Set to 0 to disable.\n   */\n  public heartbeatIncoming: number = 10000;\n\n  /**\n   * Outgoing heartbeat interval in milliseconds. Set to 0 to disable.\n   */\n  public heartbeatOutgoing: number = 10000;\n\n  /**\n   * This switches on a non standard behavior while sending WebSocket packets.\n   * It splits larger (text) packets into chunks of [maxWebSocketChunkSize]{@link Client#maxWebSocketChunkSize}.\n   * Only Java Spring brokers seems to use this mode.\n   *\n   * WebSockets, by itself, split large (text) packets,\n   * so it is not needed with a truly compliant STOMP/WebSocket broker.\n   * Actually setting it for such broker will cause large messages to fail.\n   *\n   * `false` by default.\n   *\n   * Binary frames are never split.\n   */\n  public splitLargeFrames: boolean = false;\n\n  /**\n   * See [splitLargeFrames]{@link Client#splitLargeFrames}.\n   * This has no effect if [splitLargeFrames]{@link Client#splitLargeFrames} is `false`.\n   */\n  public maxWebSocketChunkSize: number = 8 * 1024;\n\n  /**\n   * Usually the\n   * [type of WebSocket frame]{@link https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/send#Parameters}\n   * is automatically decided by type of the payload.\n   * Default is `false`, which should work with all compliant brokers.\n   *\n   * Set this flag to force binary frames.\n   */\n  public forceBinaryWSFrames: boolean = false;\n\n  /**\n   * A bug in ReactNative chops a string on occurrence of a NULL.\n   * See issue [https://github.com/stomp-js/stompjs/issues/89]{@link https://github.com/stomp-js/stompjs/issues/89}.\n   * This makes incoming WebSocket messages invalid STOMP packets.\n   * Setting this flag attempts to reverse the damage by appending a NULL.\n   * If the broker splits a large message into multiple WebSocket messages,\n   * this flag will cause data loss and abnormal termination of connection.\n   *\n   * This is not an ideal solution, but a stop gap until the underlying issue is fixed at ReactNative library.\n   */\n  public appendMissingNULLonIncoming: boolean = false;\n\n  /**\n   * Underlying WebSocket instance, READONLY.\n   */\n  get webSocket(): IStompSocket | undefined {\n    return this._stompHandler?._webSocket;\n  }\n\n  /**\n   * Connection headers, important keys - `login`, `passcode`, `host`.\n   * Though STOMP 1.2 standard marks these keys to be present, check your broker documentation for\n   * details specific to your broker.\n   */\n  public connectHeaders: StompHeaders;\n\n  /**\n   * Disconnection headers.\n   */\n  get disconnectHeaders(): StompHeaders {\n    return this._disconnectHeaders;\n  }\n\n  set disconnectHeaders(value: StompHeaders) {\n    this._disconnectHeaders = value;\n    if (this._stompHandler) {\n      this._stompHandler.disconnectHeaders = this._disconnectHeaders;\n    }\n  }\n  private _disconnectHeaders: StompHeaders;\n\n  /**\n   * This function will be called for any unhandled messages.\n   * It is useful for receiving messages sent to RabbitMQ temporary queues.\n   *\n   * It can also get invoked with stray messages while the server is processing\n   * a request to [Client#unsubscribe]{@link Client#unsubscribe}\n   * from an endpoint.\n   *\n   * The actual {@link IMessage} will be passed as parameter to the callback.\n   */\n  public onUnhandledMessage: messageCallbackType;\n\n  /**\n   * STOMP brokers can be requested to notify when an operation is actually completed.\n   * Prefer using [Client#watchForReceipt]{@link Client#watchForReceipt}. See\n   * [Client#watchForReceipt]{@link Client#watchForReceipt} for examples.\n   *\n   * The actual {@link FrameImpl} will be passed as parameter to the callback.\n   */\n  public onUnhandledReceipt: frameCallbackType;\n\n  /**\n   * Will be invoked if {@link FrameImpl} of unknown type is received from the STOMP broker.\n   *\n   * The actual {@link IFrame} will be passed as parameter to the callback.\n   */\n  public onUnhandledFrame: frameCallbackType;\n\n  /**\n   * `true` if there is a active connection with STOMP Broker\n   */\n  get connected(): boolean {\n    return !!this._stompHandler && this._stompHandler.connected;\n  }\n\n  /**\n   * Callback, invoked on before a connection connection to the STOMP broker.\n   *\n   * You can change options on the client, which will impact the immediate connect.\n   * It is valid to call [Client#decativate]{@link Client#deactivate} in this callback.\n   *\n   * As of version 5.1, this callback can be\n   * [async](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/async_function)\n   * (i.e., it can return a\n   * [Promise](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise)).\n   * In that case connect will be called only after the Promise is resolved.\n   * This can be used to reliably fetch credentials, access token etc. from some other service\n   * in an asynchronous way.\n   */\n  public beforeConnect: () => void | Promise<void>;\n\n  /**\n   * Callback, invoked on every successful connection to the STOMP broker.\n   *\n   * The actual {@link FrameImpl} will be passed as parameter to the callback.\n   * Sometimes clients will like to use headers from this frame.\n   */\n  public onConnect: frameCallbackType;\n\n  /**\n   * Callback, invoked on every successful disconnection from the STOMP broker. It will not be invoked if\n   * the STOMP broker disconnected due to an error.\n   *\n   * The actual Receipt {@link FrameImpl} acknowledging the DISCONNECT will be passed as parameter to the callback.\n   *\n   * The way STOMP protocol is designed, the connection may close/terminate without the client\n   * receiving the Receipt {@link FrameImpl} acknowledging the DISCONNECT.\n   * You might find [Client#onWebSocketClose]{@link Client#onWebSocketClose} more appropriate to watch\n   * STOMP broker disconnects.\n   */\n  public onDisconnect: frameCallbackType;\n\n  /**\n   * Callback, invoked on an ERROR frame received from the STOMP Broker.\n   * A compliant STOMP Broker will close the connection after this type of frame.\n   * Please check broker specific documentation for exact behavior.\n   *\n   * The actual {@link IFrame} will be passed as parameter to the callback.\n   */\n  public onStompError: frameCallbackType;\n\n  /**\n   * Callback, invoked when underlying WebSocket is closed.\n   *\n   * Actual [CloseEvent]{@link https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent}\n   * is passed as parameter to the callback.\n   */\n  public onWebSocketClose: closeEventCallbackType;\n\n  /**\n   * Callback, invoked when underlying WebSocket raises an error.\n   *\n   * Actual [Event]{@link https://developer.mozilla.org/en-US/docs/Web/API/Event}\n   * is passed as parameter to the callback.\n   */\n  public onWebSocketError: wsErrorCallbackType;\n\n  /**\n   * Set it to log the actual raw communication with the broker.\n   * When unset, it logs headers of the parsed frames.\n   *\n   * Change in this effects from next broker reconnect.\n   *\n   * **Caution: this assumes that frames only have valid UTF8 strings.**\n   */\n  public logRawCommunication: boolean;\n\n  /**\n   * By default, debug messages are discarded. To log to `console` following can be used:\n   *\n   * ```javascript\n   *        client.debug = function(str) {\n   *          console.log(str);\n   *        };\n   * ```\n   *\n   * Currently this method does not support levels of log. Be aware that the output can be quite verbose\n   * and may contain sensitive information (like passwords, tokens etc.).\n   */\n  public debug: debugFnType;\n\n  /**\n   * Browsers do not immediately close WebSockets when `.close` is issued.\n   * This may cause reconnection to take a longer on certain type of failures.\n   * In case of incoming heartbeat failure, this experimental flag instructs the library\n   * to discard the socket immediately (even before it is actually closed).\n   */\n  public discardWebsocketOnCommFailure: boolean = false;\n\n  /**\n   * version of STOMP protocol negotiated with the server, READONLY\n   */\n  get connectedVersion(): string | undefined {\n    return this._stompHandler ? this._stompHandler.connectedVersion : undefined;\n  }\n\n  private _stompHandler: StompHandler | undefined;\n\n  /**\n   * if the client is active (connected or going to reconnect)\n   */\n  get active(): boolean {\n    return this.state === ActivationState.ACTIVE;\n  }\n\n  /**\n   * It will be called on state change.\n   *\n   * When deactivating it may go from ACTIVE to INACTIVE without entering DEACTIVATING.\n   */\n  public onChangeState: (state: ActivationState) => void;\n\n  private _changeState(state: ActivationState) {\n    this.state = state;\n    this.onChangeState(state);\n  }\n\n  /**\n   * Activation state.\n   *\n   * It will usually be ACTIVE or INACTIVE.\n   * When deactivating it may go from ACTIVE to INACTIVE without entering DEACTIVATING.\n   */\n  public state: ActivationState = ActivationState.INACTIVE;\n\n  private _reconnector: any;\n\n  /**\n   * Create an instance.\n   */\n  constructor(conf: StompConfig = {}) {\n    // Dummy callbacks\n    const noOp = () => {};\n    this.debug = noOp;\n    this.beforeConnect = noOp;\n    this.onConnect = noOp;\n    this.onDisconnect = noOp;\n    this.onUnhandledMessage = noOp;\n    this.onUnhandledReceipt = noOp;\n    this.onUnhandledFrame = noOp;\n    this.onStompError = noOp;\n    this.onWebSocketClose = noOp;\n    this.onWebSocketError = noOp;\n    this.logRawCommunication = false;\n    this.onChangeState = noOp;\n\n    // These parameters would typically get proper values before connect is called\n    this.connectHeaders = {};\n    this._disconnectHeaders = {};\n\n    // Apply configuration\n    this.configure(conf);\n  }\n\n  /**\n   * Update configuration.\n   */\n  public configure(conf: StompConfig): void {\n    // bulk assign all properties to this\n    (Object as any).assign(this, conf);\n  }\n\n  /**\n   * Initiate the connection with the broker.\n   * If the connection breaks, as per [Client#reconnectDelay]{@link Client#reconnectDelay},\n   * it will keep trying to reconnect.\n   *\n   * Call [Client#deactivate]{@link Client#deactivate} to disconnect and stop reconnection attempts.\n   */\n  public activate(): void {\n    if (this.state === ActivationState.DEACTIVATING) {\n      this.debug(\n        'Still DEACTIVATING, please await call to deactivate before trying to re-activate'\n      );\n      throw new Error('Still DEACTIVATING, can not activate now');\n    }\n\n    if (this.active) {\n      this.debug('Already ACTIVE, ignoring request to activate');\n      return;\n    }\n\n    this._changeState(ActivationState.ACTIVE);\n\n    this._connect();\n  }\n\n  private async _connect(): Promise<void> {\n    if (this.connected) {\n      this.debug('STOMP: already connected, nothing to do');\n      return;\n    }\n\n    await this.beforeConnect();\n\n    if (!this.active) {\n      this.debug(\n        'Client has been marked inactive, will not attempt to connect'\n      );\n      return;\n    }\n\n    // setup connection watcher\n    if (this.connectionTimeout > 0) {\n      // clear first\n      if (this._connectionWatcher) {\n        clearTimeout(this._connectionWatcher);\n      }\n      this._connectionWatcher = setTimeout(() => {\n        if (this.connected) {\n          return;\n        }\n        // Connection not established, close the underlying socket\n        // a reconnection will be attempted\n        this.debug(\n          `Connection not established in ${this.connectionTimeout}ms, closing socket`\n        );\n        this.forceDisconnect();\n      }, this.connectionTimeout);\n    }\n\n    this.debug('Opening Web Socket...');\n\n    // Get the actual WebSocket (or a similar object)\n    const webSocket = this._createWebSocket();\n\n    this._stompHandler = new StompHandler(this, webSocket, {\n      debug: this.debug,\n      stompVersions: this.stompVersions,\n      connectHeaders: this.connectHeaders,\n      disconnectHeaders: this._disconnectHeaders,\n      heartbeatIncoming: this.heartbeatIncoming,\n      heartbeatOutgoing: this.heartbeatOutgoing,\n      splitLargeFrames: this.splitLargeFrames,\n      maxWebSocketChunkSize: this.maxWebSocketChunkSize,\n      forceBinaryWSFrames: this.forceBinaryWSFrames,\n      logRawCommunication: this.logRawCommunication,\n      appendMissingNULLonIncoming: this.appendMissingNULLonIncoming,\n      discardWebsocketOnCommFailure: this.discardWebsocketOnCommFailure,\n\n      onConnect: frame => {\n        // Successfully connected, stop the connection watcher\n        if (this._connectionWatcher) {\n          clearTimeout(this._connectionWatcher);\n          this._connectionWatcher = undefined;\n        }\n\n        if (!this.active) {\n          this.debug(\n            'STOMP got connected while deactivate was issued, will disconnect now'\n          );\n          this._disposeStompHandler();\n          return;\n        }\n        this.onConnect(frame);\n      },\n      onDisconnect: frame => {\n        this.onDisconnect(frame);\n      },\n      onStompError: frame => {\n        this.onStompError(frame);\n      },\n      onWebSocketClose: evt => {\n        this._stompHandler = undefined; // a new one will be created in case of a reconnect\n\n        if (this.state === ActivationState.DEACTIVATING) {\n          // Mark deactivation complete\n          this._changeState(ActivationState.INACTIVE);\n        }\n\n        // The callback is called before attempting to reconnect, this would allow the client\n        // to be `deactivated` in the callback.\n        this.onWebSocketClose(evt);\n\n        if (this.active) {\n          this._schedule_reconnect();\n        }\n      },\n      onWebSocketError: evt => {\n        this.onWebSocketError(evt);\n      },\n      onUnhandledMessage: message => {\n        this.onUnhandledMessage(message);\n      },\n      onUnhandledReceipt: frame => {\n        this.onUnhandledReceipt(frame);\n      },\n      onUnhandledFrame: frame => {\n        this.onUnhandledFrame(frame);\n      },\n    });\n\n    this._stompHandler.start();\n  }\n\n  private _createWebSocket(): IStompSocket {\n    let webSocket: IStompSocket;\n\n    if (this.webSocketFactory) {\n      webSocket = this.webSocketFactory();\n    } else if (this.brokerURL) {\n      webSocket = new WebSocket(\n        this.brokerURL,\n        this.stompVersions.protocolVersions()\n      );\n    } else {\n      throw new Error('Either brokerURL or webSocketFactory must be provided');\n    }\n    webSocket.binaryType = 'arraybuffer';\n    return webSocket;\n  }\n\n  private _schedule_reconnect(): void {\n    if (this.reconnectDelay > 0) {\n      this.debug(`STOMP: scheduling reconnection in ${this.reconnectDelay}ms`);\n\n      this._reconnector = setTimeout(() => {\n        this._connect();\n      }, this.reconnectDelay);\n    }\n  }\n\n  /**\n   * Disconnect if connected and stop auto reconnect loop.\n   * Appropriate callbacks will be invoked if there is an underlying STOMP connection.\n   *\n   * This call is async. It will resolve immediately if there is no underlying active websocket,\n   * otherwise, it will resolve after the underlying websocket is properly disposed of.\n   *\n   * It is not an error to invoke this method more than once.\n   * Each of those would resolve on completion of deactivation.\n   *\n   * To reactivate, you can call [Client#activate]{@link Client#activate}.\n   *\n   * Experimental: pass `force: true` to immediately discard the underlying connection.\n   * This mode will skip both the STOMP and the Websocket shutdown sequences.\n   * In some cases, browsers take a long time in the Websocket shutdown if the underlying connection had gone stale.\n   * Using this mode can speed up.\n   * When this mode is used, the actual Websocket may linger for a while\n   * and the broker may not realize that the connection is no longer in use.\n   *\n   * It is possible to invoke this method initially without the `force` option\n   * and subsequently, say after a wait, with the `force` option.\n   */\n  public async deactivate(options: { force?: boolean } = {}): Promise<void> {\n    const force: boolean = options.force || false;\n    const needToDispose = this.active;\n    let retPromise: Promise<void>;\n\n    if (this.state === ActivationState.INACTIVE) {\n      this.debug(`Already INACTIVE, nothing more to do`);\n      return Promise.resolve();\n    }\n\n    this._changeState(ActivationState.DEACTIVATING);\n\n    // Clear if a reconnection was scheduled\n    if (this._reconnector) {\n      clearTimeout(this._reconnector);\n      this._reconnector = undefined;\n    }\n\n    if (\n      this._stompHandler &&\n      // @ts-ignore - if there is a _stompHandler, there is the webSocket\n      this.webSocket.readyState !== StompSocketState.CLOSED\n    ) {\n      const origOnWebSocketClose = this._stompHandler.onWebSocketClose;\n      // we need to wait for the underlying websocket to close\n      retPromise = new Promise<void>((resolve, reject) => {\n        // @ts-ignore - there is a _stompHandler\n        this._stompHandler.onWebSocketClose = evt => {\n          origOnWebSocketClose(evt);\n          resolve();\n        };\n      });\n    } else {\n      // indicate that auto reconnect loop should terminate\n      this._changeState(ActivationState.INACTIVE);\n      return Promise.resolve();\n    }\n\n    if (force) {\n      this._stompHandler?.discardWebsocket();\n    } else if (needToDispose) {\n      this._disposeStompHandler();\n    }\n\n    return retPromise;\n  }\n\n  /**\n   * Force disconnect if there is an active connection by directly closing the underlying WebSocket.\n   * This is different than a normal disconnect where a DISCONNECT sequence is carried out with the broker.\n   * After forcing disconnect, automatic reconnect will be attempted.\n   * To stop further reconnects call [Client#deactivate]{@link Client#deactivate} as well.\n   */\n  public forceDisconnect() {\n    if (this._stompHandler) {\n      this._stompHandler.forceDisconnect();\n    }\n  }\n\n  private _disposeStompHandler() {\n    // Dispose STOMP Handler\n    if (this._stompHandler) {\n      this._stompHandler.dispose();\n    }\n  }\n\n  /**\n   * Send a message to a named destination. Refer to your STOMP broker documentation for types\n   * and naming of destinations.\n   *\n   * STOMP protocol specifies and suggests some headers and also allows broker specific headers.\n   *\n   * `body` must be String.\n   * You will need to covert the payload to string in case it is not string (e.g. JSON).\n   *\n   * To send a binary message body use binaryBody parameter. It should be a\n   * [Uint8Array](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Uint8Array).\n   * Sometimes brokers may not support binary frames out of the box.\n   * Please check your broker documentation.\n   *\n   * `content-length` header is automatically added to the STOMP Frame sent to the broker.\n   * Set `skipContentLengthHeader` to indicate that `content-length` header should not be added.\n   * For binary messages `content-length` header is always added.\n   *\n   * Caution: The broker will, most likely, report an error and disconnect if message body has NULL octet(s)\n   * and `content-length` header is missing.\n   *\n   * ```javascript\n   *        client.publish({destination: \"/queue/test\", headers: {priority: 9}, body: \"Hello, STOMP\"});\n   *\n   *        // Only destination is mandatory parameter\n   *        client.publish({destination: \"/queue/test\", body: \"Hello, STOMP\"});\n   *\n   *        // Skip content-length header in the frame to the broker\n   *        client.publish({\"/queue/test\", body: \"Hello, STOMP\", skipContentLengthHeader: true});\n   *\n   *        var binaryData = generateBinaryData(); // This need to be of type Uint8Array\n   *        // setting content-type header is not mandatory, however a good practice\n   *        client.publish({destination: '/topic/special', binaryBody: binaryData,\n   *                         headers: {'content-type': 'application/octet-stream'}});\n   * ```\n   */\n  public publish(params: IPublishParams) {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    this._stompHandler.publish(params);\n  }\n\n  private _checkConnection() {\n    if (!this.connected) {\n      throw new TypeError('There is no underlying STOMP connection');\n    }\n  }\n\n  /**\n   * STOMP brokers may carry out operation asynchronously and allow requesting for acknowledgement.\n   * To request an acknowledgement, a `receipt` header needs to be sent with the actual request.\n   * The value (say receipt-id) for this header needs to be unique for each use. Typically a sequence, a UUID, a\n   * random number or a combination may be used.\n   *\n   * A complaint broker will send a RECEIPT frame when an operation has actually been completed.\n   * The operation needs to be matched based in the value of the receipt-id.\n   *\n   * This method allow watching for a receipt and invoke the callback\n   * when corresponding receipt has been received.\n   *\n   * The actual {@link FrameImpl} will be passed as parameter to the callback.\n   *\n   * Example:\n   * ```javascript\n   *        // Subscribing with acknowledgement\n   *        let receiptId = randomText();\n   *\n   *        client.watchForReceipt(receiptId, function() {\n   *          // Will be called after server acknowledges\n   *        });\n   *\n   *        client.subscribe(TEST.destination, onMessage, {receipt: receiptId});\n   *\n   *\n   *        // Publishing with acknowledgement\n   *        receiptId = randomText();\n   *\n   *        client.watchForReceipt(receiptId, function() {\n   *          // Will be called after server acknowledges\n   *        });\n   *        client.publish({destination: TEST.destination, headers: {receipt: receiptId}, body: msg});\n   * ```\n   */\n  public watchForReceipt(receiptId: string, callback: frameCallbackType): void {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    this._stompHandler.watchForReceipt(receiptId, callback);\n  }\n\n  /**\n   * Subscribe to a STOMP Broker location. The callback will be invoked for each received message with\n   * the {@link IMessage} as argument.\n   *\n   * Note: The library will generate an unique ID if there is none provided in the headers.\n   *       To use your own ID, pass it using the headers argument.\n   *\n   * ```javascript\n   *        callback = function(message) {\n   *        // called when the client receives a STOMP message from the server\n   *          if (message.body) {\n   *            alert(\"got message with body \" + message.body)\n   *          } else {\n   *            alert(\"got empty message\");\n   *          }\n   *        });\n   *\n   *        var subscription = client.subscribe(\"/queue/test\", callback);\n   *\n   *        // Explicit subscription id\n   *        var mySubId = 'my-subscription-id-001';\n   *        var subscription = client.subscribe(destination, callback, { id: mySubId });\n   * ```\n   */\n  public subscribe(\n    destination: string,\n    callback: messageCallbackType,\n    headers: StompHeaders = {}\n  ): StompSubscription {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    return this._stompHandler.subscribe(destination, callback, headers);\n  }\n\n  /**\n   * It is preferable to unsubscribe from a subscription by calling\n   * `unsubscribe()` directly on {@link StompSubscription} returned by `client.subscribe()`:\n   *\n   * ```javascript\n   *        var subscription = client.subscribe(destination, onmessage);\n   *        // ...\n   *        subscription.unsubscribe();\n   * ```\n   *\n   * See: http://stomp.github.com/stomp-specification-1.2.html#UNSUBSCRIBE UNSUBSCRIBE Frame\n   */\n  public unsubscribe(id: string, headers: StompHeaders = {}): void {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    this._stompHandler.unsubscribe(id, headers);\n  }\n\n  /**\n   * Start a transaction, the returned {@link ITransaction} has methods - [commit]{@link ITransaction#commit}\n   * and [abort]{@link ITransaction#abort}.\n   *\n   * `transactionId` is optional, if not passed the library will generate it internally.\n   */\n  public begin(transactionId?: string): ITransaction {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    return this._stompHandler.begin(transactionId);\n  }\n\n  /**\n   * Commit a transaction.\n   *\n   * It is preferable to commit a transaction by calling [commit]{@link ITransaction#commit} directly on\n   * {@link ITransaction} returned by [client.begin]{@link Client#begin}.\n   *\n   * ```javascript\n   *        var tx = client.begin(txId);\n   *        //...\n   *        tx.commit();\n   * ```\n   */\n  public commit(transactionId: string): void {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    this._stompHandler.commit(transactionId);\n  }\n\n  /**\n   * Abort a transaction.\n   * It is preferable to abort a transaction by calling [abort]{@link ITransaction#abort} directly on\n   * {@link ITransaction} returned by [client.begin]{@link Client#begin}.\n   *\n   * ```javascript\n   *        var tx = client.begin(txId);\n   *        //...\n   *        tx.abort();\n   * ```\n   */\n  public abort(transactionId: string): void {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    this._stompHandler.abort(transactionId);\n  }\n\n  /**\n   * ACK a message. It is preferable to acknowledge a message by calling [ack]{@link IMessage#ack} directly\n   * on the {@link IMessage} handled by a subscription callback:\n   *\n   * ```javascript\n   *        var callback = function (message) {\n   *          // process the message\n   *          // acknowledge it\n   *          message.ack();\n   *        };\n   *        client.subscribe(destination, callback, {'ack': 'client'});\n   * ```\n   */\n  public ack(\n    messageId: string,\n    subscriptionId: string,\n    headers: StompHeaders = {}\n  ): void {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    this._stompHandler.ack(messageId, subscriptionId, headers);\n  }\n\n  /**\n   * NACK a message. It is preferable to acknowledge a message by calling [nack]{@link IMessage#nack} directly\n   * on the {@link IMessage} handled by a subscription callback:\n   *\n   * ```javascript\n   *        var callback = function (message) {\n   *          // process the message\n   *          // an error occurs, nack it\n   *          message.nack();\n   *        };\n   *        client.subscribe(destination, callback, {'ack': 'client'});\n   * ```\n   */\n  public nack(\n    messageId: string,\n    subscriptionId: string,\n    headers: StompHeaders = {}\n  ): void {\n    this._checkConnection();\n    // @ts-ignore - we already checked that there is a _stompHandler, and it is connected\n    this._stompHandler.nack(messageId, subscriptionId, headers);\n  }\n}\n","import { Client } from '../client';\nimport { StompHeaders } from '../stomp-headers';\nimport { frameCallbackType, messageCallbackType } from '../types';\nimport { HeartbeatInfo } from './heartbeat-info';\n\n/**\n * Available for backward compatibility, please shift to using {@link Client}.\n *\n * **Deprecated**\n *\n * Part of `@stomp/stompjs`.\n *\n * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)\n */\nexport class CompatClient extends Client {\n  /**\n   * It is no op now. No longer needed. Large packets work out of the box.\n   */\n  public maxWebSocketFrameSize: number = 16 * 1024;\n\n  /**\n   * Available for backward compatibility, please shift to using {@link Client}\n   * and [Client#webSocketFactory]{@link Client#webSocketFactory}.\n   *\n   * **Deprecated**\n   *\n   * @internal\n   */\n  constructor(webSocketFactory: () => any) {\n    super();\n    this.reconnect_delay = 0;\n    this.webSocketFactory = webSocketFactory;\n    // Default from previous version\n    this.debug = (...message: any[]) => {\n      console.log(...message);\n    };\n  }\n\n  private _parseConnect(...args: any[]): any {\n    let closeEventCallback;\n    let connectCallback;\n    let errorCallback;\n    let headers: StompHeaders = {};\n    if (args.length < 2) {\n      throw new Error('Connect requires at least 2 arguments');\n    }\n    if (typeof args[1] === 'function') {\n      [headers, connectCallback, errorCallback, closeEventCallback] = args;\n    } else {\n      switch (args.length) {\n        case 6:\n          [\n            headers.login,\n            headers.passcode,\n            connectCallback,\n            errorCallback,\n            closeEventCallback,\n            headers.host,\n          ] = args;\n          break;\n        default:\n          [\n            headers.login,\n            headers.passcode,\n            connectCallback,\n            errorCallback,\n            closeEventCallback,\n          ] = args;\n      }\n    }\n\n    return [headers, connectCallback, errorCallback, closeEventCallback];\n  }\n\n  /**\n   * Available for backward compatibility, please shift to using [Client#activate]{@link Client#activate}.\n   *\n   * **Deprecated**\n   *\n   * The `connect` method accepts different number of arguments and types. See the Overloads list. Use the\n   * version with headers to pass your broker specific options.\n   *\n   * overloads:\n   * - connect(headers, connectCallback)\n   * - connect(headers, connectCallback, errorCallback)\n   * - connect(login, passcode, connectCallback)\n   * - connect(login, passcode, connectCallback, errorCallback)\n   * - connect(login, passcode, connectCallback, errorCallback, closeEventCallback)\n   * - connect(login, passcode, connectCallback, errorCallback, closeEventCallback, host)\n   *\n   * params:\n   * - headers, see [Client#connectHeaders]{@link Client#connectHeaders}\n   * - connectCallback, see [Client#onConnect]{@link Client#onConnect}\n   * - errorCallback, see [Client#onStompError]{@link Client#onStompError}\n   * - closeEventCallback, see [Client#onWebSocketClose]{@link Client#onWebSocketClose}\n   * - login [String], see [Client#connectHeaders](../classes/Client.html#connectHeaders)\n   * - passcode [String], [Client#connectHeaders](../classes/Client.html#connectHeaders)\n   * - host [String], see [Client#connectHeaders](../classes/Client.html#connectHeaders)\n   *\n   * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)\n   */\n  public connect(...args: any[]): void {\n    const out = this._parseConnect(...args);\n\n    if (out[0]) {\n      this.connectHeaders = out[0];\n    }\n    if (out[1]) {\n      this.onConnect = out[1];\n    }\n    if (out[2]) {\n      this.onStompError = out[2];\n    }\n    if (out[3]) {\n      this.onWebSocketClose = out[3];\n    }\n\n    super.activate();\n  }\n\n  /**\n   * Available for backward compatibility, please shift to using [Client#deactivate]{@link Client#deactivate}.\n   *\n   * **Deprecated**\n   *\n   * See:\n   * [Client#onDisconnect]{@link Client#onDisconnect}, and\n   * [Client#disconnectHeaders]{@link Client#disconnectHeaders}\n   *\n   * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)\n   */\n  public disconnect(\n    disconnectCallback?: any,\n    headers: StompHeaders = {}\n  ): void {\n    if (disconnectCallback) {\n      this.onDisconnect = disconnectCallback;\n    }\n    this.disconnectHeaders = headers;\n\n    super.deactivate();\n  }\n\n  /**\n   * Available for backward compatibility, use [Client#publish]{@link Client#publish}.\n   *\n   * Send a message to a named destination. Refer to your STOMP broker documentation for types\n   * and naming of destinations. The headers will, typically, be available to the subscriber.\n   * However, there may be special purpose headers corresponding to your STOMP broker.\n   *\n   *  **Deprecated**, use [Client#publish]{@link Client#publish}\n   *\n   * Note: Body must be String. You will need to covert the payload to string in case it is not string (e.g. JSON)\n   *\n   * ```javascript\n   *        client.send(\"/queue/test\", {priority: 9}, \"Hello, STOMP\");\n   *\n   *        // If you want to send a message with a body, you must also pass the headers argument.\n   *        client.send(\"/queue/test\", {}, \"Hello, STOMP\");\n   * ```\n   *\n   * To upgrade, please follow the [Upgrade Guide](../additional-documentation/upgrading.html)\n   */\n  public send(\n    destination: string,\n    headers: { [key: string]: any } = {},\n    body: string = ''\n  ): void {\n    headers = (Object as any).assign({}, headers);\n\n    const skipContentLengthHeader = headers['content-length'] === false;\n    if (skipContentLengthHeader) {\n      delete headers['content-length'];\n    }\n    this.publish({\n      destination,\n      headers: headers as StompHeaders,\n      body,\n      skipContentLengthHeader,\n    });\n  }\n\n  /**\n   * Available for backward compatibility, renamed to [Client#reconnectDelay]{@link Client#reconnectDelay}.\n   *\n   * **Deprecated**\n   */\n  set reconnect_delay(value: number) {\n    this.reconnectDelay = value;\n  }\n\n  /**\n   * Available for backward compatibility, renamed to [Client#webSocket]{@link Client#webSocket}.\n   *\n   * **Deprecated**\n   */\n  get ws(): any {\n    return this.webSocket;\n  }\n\n  /**\n   * Available for backward compatibility, renamed to [Client#connectedVersion]{@link Client#connectedVersion}.\n   *\n   * **Deprecated**\n   */\n  get version() {\n    return this.connectedVersion;\n  }\n\n  /**\n   * Available for backward compatibility, renamed to [Client#onUnhandledMessage]{@link Client#onUnhandledMessage}.\n   *\n   * **Deprecated**\n   */\n  get onreceive(): messageCallbackType {\n    return this.onUnhandledMessage;\n  }\n\n  /**\n   * Available for backward compatibility, renamed to [Client#onUnhandledMessage]{@link Client#onUnhandledMessage}.\n   *\n   * **Deprecated**\n   */\n  set onreceive(value: messageCallbackType) {\n    this.onUnhandledMessage = value;\n  }\n\n  /**\n   * Available for backward compatibility, renamed to [Client#onUnhandledReceipt]{@link Client#onUnhandledReceipt}.\n   * Prefer using [Client#watchForReceipt]{@link Client#watchForReceipt}.\n   *\n   * **Deprecated**\n   */\n  get onreceipt(): frameCallbackType {\n    return this.onUnhandledReceipt;\n  }\n\n  /**\n   * Available for backward compatibility, renamed to [Client#onUnhandledReceipt]{@link Client#onUnhandledReceipt}.\n   *\n   * **Deprecated**\n   */\n  set onreceipt(value: frameCallbackType) {\n    this.onUnhandledReceipt = value;\n  }\n\n  private _heartbeatInfo: HeartbeatInfo = new HeartbeatInfo(this);\n\n  /**\n   * Available for backward compatibility, renamed to [Client#heartbeatIncoming]{@link Client#heartbeatIncoming}\n   * [Client#heartbeatOutgoing]{@link Client#heartbeatOutgoing}.\n   *\n   * **Deprecated**\n   */\n  get heartbeat() {\n    return this._heartbeatInfo;\n  }\n\n  /**\n   * Available for backward compatibility, renamed to [Client#heartbeatIncoming]{@link Client#heartbeatIncoming}\n   * [Client#heartbeatOutgoing]{@link Client#heartbeatOutgoing}.\n   *\n   * **Deprecated**\n   */\n  set heartbeat(value: { incoming: number; outgoing: number }) {\n    this.heartbeatIncoming = value.incoming;\n    this.heartbeatOutgoing = value.outgoing;\n  }\n}\n","import { CompatClient } from './compat-client';\n\n/**\n * Part of `@stomp/stompjs`.\n *\n * @internal\n */\nexport class HeartbeatInfo {\n  constructor(private client: CompatClient) {}\n\n  get outgoing(): number {\n    return this.client.heartbeatOutgoing;\n  }\n\n  set outgoing(value: number) {\n    this.client.heartbeatOutgoing = value;\n  }\n\n  get incoming(): number {\n    return this.client.heartbeatIncoming;\n  }\n\n  set incoming(value: number) {\n    this.client.heartbeatIncoming = value;\n  }\n}\n","import { Versions } from '../versions';\nimport { CompatClient } from './compat-client';\nimport { IStompSocket } from '../types';\n\n/**\n * @internal\n */\ndeclare const WebSocket: {\n  prototype: IStompSocket;\n  new (url: string, protocols?: string | string[]): IStompSocket;\n};\n\n/**\n * STOMP Class, acts like a factory to create {@link Client}.\n *\n * Part of `@stomp/stompjs`.\n *\n * **Deprecated**\n *\n * It will be removed in next major version. Please switch to {@link Client}.\n */\nexport class Stomp {\n  /**\n   * In case you need to use a non standard class for WebSocket.\n   *\n   * For example when using within NodeJS environment:\n   *\n   * ```javascript\n   *        StompJs = require('../../esm5/');\n   *        Stomp = StompJs.Stomp;\n   *        Stomp.WebSocketClass = require('websocket').w3cwebsocket;\n   * ```\n   *\n   * **Deprecated**\n   *\n   *\n   * It will be removed in next major version. Please switch to {@link Client}\n   * using [Client#webSocketFactory]{@link Client#webSocketFactory}.\n   */\n  // tslint:disable-next-line:variable-name\n  public static WebSocketClass: any = null;\n\n  /**\n   * This method creates a WebSocket client that is connected to\n   * the STOMP server located at the url.\n   *\n   * ```javascript\n   *        var url = \"ws://localhost:61614/stomp\";\n   *        var client = Stomp.client(url);\n   * ```\n   *\n   * **Deprecated**\n   *\n   * It will be removed in next major version. Please switch to {@link Client}\n   * using [Client#brokerURL]{@link Client#brokerURL}.\n   */\n  public static client(url: string, protocols?: string[]): CompatClient {\n    // This is a hack to allow another implementation than the standard\n    // HTML5 WebSocket class.\n    //\n    // It is possible to use another class by calling\n    //\n    //     Stomp.WebSocketClass = MozWebSocket\n    //\n    // *prior* to call `Stomp.client()`.\n    //\n    // This hack is deprecated and `Stomp.over()` method should be used\n    // instead.\n\n    // See remarks on the function Stomp.over\n    if (protocols == null) {\n      protocols = Versions.default.protocolVersions();\n    }\n    const wsFn = () => {\n      const klass = Stomp.WebSocketClass || WebSocket;\n      return new klass(url, protocols);\n    };\n\n    return new CompatClient(wsFn);\n  }\n\n  /**\n   * This method is an alternative to [Stomp#client]{@link Stomp#client} to let the user\n   * specify the WebSocket to use (either a standard HTML5 WebSocket or\n   * a similar object).\n   *\n   * In order to support reconnection, the function Client._connect should be callable more than once.\n   * While reconnecting\n   * a new instance of underlying transport (TCP Socket, WebSocket or SockJS) will be needed. So, this function\n   * alternatively allows passing a function that should return a new instance of the underlying socket.\n   *\n   * ```javascript\n   *        var client = Stomp.over(function(){\n   *          return new WebSocket('ws://localhost:15674/ws')\n   *        });\n   * ```\n   *\n   * **Deprecated**\n   *\n   * It will be removed in next major version. Please switch to {@link Client}\n   * using [Client#webSocketFactory]{@link Client#webSocketFactory}.\n   */\n  public static over(ws: any): CompatClient {\n    let wsFn: () => any;\n\n    if (typeof ws === 'function') {\n      wsFn = ws;\n    } else {\n      console.warn(\n        'Stomp.over did not receive a factory, auto reconnect will not work. ' +\n          'Please see https://stomp-js.github.io/api-docs/latest/classes/Stomp.html#over'\n      );\n      wsFn = () => ws;\n    }\n\n    return new CompatClient(wsFn);\n  }\n}\n","import { BYTE } from './byte';\nimport { IFrame } from './i-frame';\nimport { StompHeaders } from './stomp-headers';\nimport { IRawFrameType } from './types';\n\n/**\n * Frame class represents a STOMP frame.\n *\n * @internal\n */\nexport class FrameImpl implements IFrame {\n  /**\n   * STOMP Command\n   */\n  public command: string;\n\n  /**\n   * Headers, key value pairs.\n   */\n  public headers: StompHeaders;\n\n  /**\n   * Is this frame binary (based on whether body/binaryBody was passed when creating this frame).\n   */\n  public isBinaryBody: boolean;\n\n  /**\n   * body of the frame\n   */\n  get body(): string {\n    if (!this._body && this.isBinaryBody) {\n      this._body = new TextDecoder().decode(this._binaryBody);\n    }\n    return this._body || '';\n  }\n  private _body: string | undefined;\n\n  /**\n   * body as Uint8Array\n   */\n  get binaryBody(): Uint8Array {\n    if (!this._binaryBody && !this.isBinaryBody) {\n      this._binaryBody = new TextEncoder().encode(this._body);\n    }\n    // At this stage it will definitely have a valid value\n    return this._binaryBody as Uint8Array;\n  }\n  private _binaryBody: Uint8Array | undefined;\n\n  private escapeHeaderValues: boolean;\n  private skipContentLengthHeader: boolean;\n\n  /**\n   * Frame constructor. `command`, `headers` and `body` are available as properties.\n   *\n   * @internal\n   */\n  constructor(params: {\n    command: string;\n    headers?: StompHeaders;\n    body?: string;\n    binaryBody?: Uint8Array;\n    escapeHeaderValues?: boolean;\n    skipContentLengthHeader?: boolean;\n  }) {\n    const {\n      command,\n      headers,\n      body,\n      binaryBody,\n      escapeHeaderValues,\n      skipContentLengthHeader,\n    } = params;\n    this.command = command;\n    this.headers = (Object as any).assign({}, headers || {});\n\n    if (binaryBody) {\n      this._binaryBody = binaryBody;\n      this.isBinaryBody = true;\n    } else {\n      this._body = body || '';\n      this.isBinaryBody = false;\n    }\n    this.escapeHeaderValues = escapeHeaderValues || false;\n    this.skipContentLengthHeader = skipContentLengthHeader || false;\n  }\n\n  /**\n   * deserialize a STOMP Frame from raw data.\n   *\n   * @internal\n   */\n  public static fromRawFrame(\n    rawFrame: IRawFrameType,\n    escapeHeaderValues: boolean\n  ): FrameImpl {\n    const headers: StompHeaders = {};\n    const trim = (str: string): string => str.replace(/^\\s+|\\s+$/g, '');\n\n    // In case of repeated headers, as per standards, first value need to be used\n    for (const header of rawFrame.headers.reverse()) {\n      const idx = header.indexOf(':');\n\n      const key = trim(header[0]);\n      let value = trim(header[1]);\n\n      if (\n        escapeHeaderValues &&\n        rawFrame.command !== 'CONNECT' &&\n        rawFrame.command !== 'CONNECTED'\n      ) {\n        value = FrameImpl.hdrValueUnEscape(value);\n      }\n\n      headers[key] = value;\n    }\n\n    return new FrameImpl({\n      command: rawFrame.command as string,\n      headers,\n      binaryBody: rawFrame.binaryBody,\n      escapeHeaderValues,\n    });\n  }\n\n  /**\n   * @internal\n   */\n  public toString(): string {\n    return this.serializeCmdAndHeaders();\n  }\n\n  /**\n   * serialize this Frame in a format suitable to be passed to WebSocket.\n   * If the body is string the output will be string.\n   * If the body is binary (i.e. of type Unit8Array) it will be serialized to ArrayBuffer.\n   *\n   * @internal\n   */\n  public serialize(): string | ArrayBuffer {\n    const cmdAndHeaders = this.serializeCmdAndHeaders();\n\n    if (this.isBinaryBody) {\n      return FrameImpl.toUnit8Array(\n        cmdAndHeaders,\n        this._binaryBody as Uint8Array\n      ).buffer;\n    } else {\n      return cmdAndHeaders + this._body + BYTE.NULL;\n    }\n  }\n\n  private serializeCmdAndHeaders(): string {\n    const lines = [this.command];\n    if (this.skipContentLengthHeader) {\n      delete this.headers['content-length'];\n    }\n\n    for (const name of Object.keys(this.headers || {})) {\n      const value = this.headers[name];\n      if (\n        this.escapeHeaderValues &&\n        this.command !== 'CONNECT' &&\n        this.command !== 'CONNECTED'\n      ) {\n        lines.push(`${name}:${FrameImpl.hdrValueEscape(`${value}`)}`);\n      } else {\n        lines.push(`${name}:${value}`);\n      }\n    }\n    if (\n      this.isBinaryBody ||\n      (!this.isBodyEmpty() && !this.skipContentLengthHeader)\n    ) {\n      lines.push(`content-length:${this.bodyLength()}`);\n    }\n    return lines.join(BYTE.LF) + BYTE.LF + BYTE.LF;\n  }\n\n  private isBodyEmpty(): boolean {\n    return this.bodyLength() === 0;\n  }\n\n  private bodyLength(): number {\n    const binaryBody = this.binaryBody;\n    return binaryBody ? binaryBody.length : 0;\n  }\n\n  /**\n   * Compute the size of a UTF-8 string by counting its number of bytes\n   * (and not the number of characters composing the string)\n   */\n  private static sizeOfUTF8(s: string): number {\n    return s ? new TextEncoder().encode(s).length : 0;\n  }\n\n  private static toUnit8Array(\n    cmdAndHeaders: string,\n    binaryBody: Uint8Array\n  ): Uint8Array {\n    const uint8CmdAndHeaders = new TextEncoder().encode(cmdAndHeaders);\n    const nullTerminator = new Uint8Array([0]);\n    const uint8Frame = new Uint8Array(\n      uint8CmdAndHeaders.length + binaryBody.length + nullTerminator.length\n    );\n\n    uint8Frame.set(uint8CmdAndHeaders);\n    uint8Frame.set(binaryBody, uint8CmdAndHeaders.length);\n    uint8Frame.set(\n      nullTerminator,\n      uint8CmdAndHeaders.length + binaryBody.length\n    );\n\n    return uint8Frame;\n  }\n  /**\n   * Serialize a STOMP frame as per STOMP standards, suitable to be sent to the STOMP broker.\n   *\n   * @internal\n   */\n  public static marshall(params: {\n    command: string;\n    headers?: StompHeaders;\n    body?: string;\n    binaryBody?: Uint8Array;\n    escapeHeaderValues?: boolean;\n    skipContentLengthHeader?: boolean;\n  }) {\n    const frame = new FrameImpl(params);\n    return frame.serialize();\n  }\n\n  /**\n   *  Escape header values\n   */\n  private static hdrValueEscape(str: string): string {\n    return str\n      .replace(/\\\\/g, '\\\\\\\\')\n      .replace(/\\r/g, '\\\\r')\n      .replace(/\\n/g, '\\\\n')\n      .replace(/:/g, '\\\\c');\n  }\n\n  /**\n   * UnEscape header values\n   */\n  private static hdrValueUnEscape(str: string): string {\n    return str\n      .replace(/\\\\r/g, '\\r')\n      .replace(/\\\\n/g, '\\n')\n      .replace(/\\\\c/g, ':')\n      .replace(/\\\\\\\\/g, '\\\\');\n  }\n}\n","import { IRawFrameType } from './types';\n\n/**\n * @internal\n */\nconst NULL = 0;\n/**\n * @internal\n */\nconst LF = 10;\n/**\n * @internal\n */\nconst CR = 13;\n/**\n * @internal\n */\nconst COLON = 58;\n\n/**\n * This is an evented, rec descent parser.\n * A stream of Octets can be passed and whenever it recognizes\n * a complete Frame or an incoming ping it will invoke the registered callbacks.\n *\n * All incoming Octets are fed into _onByte function.\n * Depending on current state the _onByte function keeps changing.\n * Depending on the state it keeps accumulating into _token and _results.\n * State is indicated by current value of _onByte, all states are named as _collect.\n *\n * STOMP standards https://stomp.github.io/stomp-specification-1.2.html\n * imply that all lengths are considered in bytes (instead of string lengths).\n * So, before actual parsing, if the incoming data is String it is converted to Octets.\n * This allows faithful implementation of the protocol and allows NULL Octets to be present in the body.\n *\n * There is no peek function on the incoming data.\n * When a state change occurs based on an Octet without consuming the Octet,\n * the Octet, after state change, is fed again (_reinjectByte).\n * This became possible as the state change can be determined by inspecting just one Octet.\n *\n * There are two modes to collect the body, if content-length header is there then it by counting Octets\n * otherwise it is determined by NULL terminator.\n *\n * Following the standards, the command and headers are converted to Strings\n * and the body is returned as Octets.\n * Headers are returned as an array and not as Hash - to allow multiple occurrence of an header.\n *\n * This parser does not use Regular Expressions as that can only operate on Strings.\n *\n * It handles if multiple STOMP frames are given as one chunk, a frame is split into multiple chunks, or\n * any combination there of. The parser remembers its state (any partial frame) and continues when a new chunk\n * is pushed.\n *\n * Typically the higher level function will convert headers to Hash, handle unescaping of header values\n * (which is protocol version specific), and convert body to text.\n *\n * Check the parser.spec.js to understand cases that this parser is supposed to handle.\n *\n * Part of `@stomp/stompjs`.\n *\n * @internal\n */\nexport class Parser {\n  private readonly _encoder = new TextEncoder();\n  private readonly _decoder = new TextDecoder();\n\n  // @ts-ignore - it always has a value\n  private _results: IRawFrameType;\n\n  private _token: number[] = [];\n  private _headerKey: string | undefined;\n  private _bodyBytesRemaining: number | undefined;\n\n  // @ts-ignore - it always has a value\n  private _onByte: (byte: number) => void;\n\n  public constructor(\n    public onFrame: (rawFrame: IRawFrameType) => void,\n    public onIncomingPing: () => void\n  ) {\n    this._initState();\n  }\n\n  public parseChunk(\n    segment: string | ArrayBuffer,\n    appendMissingNULLonIncoming: boolean = false\n  ) {\n    let chunk: Uint8Array;\n\n    if (segment instanceof ArrayBuffer) {\n      chunk = new Uint8Array(segment);\n    } else {\n      chunk = this._encoder.encode(segment);\n    }\n\n    // See https://github.com/stomp-js/stompjs/issues/89\n    // Remove when underlying issue is fixed.\n    //\n    // Send a NULL byte, if the last byte of a Text frame was not NULL.F\n    if (appendMissingNULLonIncoming && chunk[chunk.length - 1] !== 0) {\n      const chunkWithNull = new Uint8Array(chunk.length + 1);\n      chunkWithNull.set(chunk, 0);\n      chunkWithNull[chunk.length] = 0;\n      chunk = chunkWithNull;\n    }\n\n    // tslint:disable-next-line:prefer-for-of\n    for (let i = 0; i < chunk.length; i++) {\n      const byte = chunk[i];\n      this._onByte(byte);\n    }\n  }\n\n  // The following implements a simple Rec Descent Parser.\n  // The grammar is simple and just one byte tells what should be the next state\n\n  private _collectFrame(byte: number): void {\n    if (byte === NULL) {\n      // Ignore\n      return;\n    }\n    if (byte === CR) {\n      // Ignore CR\n      return;\n    }\n    if (byte === LF) {\n      // Incoming Ping\n      this.onIncomingPing();\n      return;\n    }\n\n    this._onByte = this._collectCommand;\n    this._reinjectByte(byte);\n  }\n\n  private _collectCommand(byte: number): void {\n    if (byte === CR) {\n      // Ignore CR\n      return;\n    }\n    if (byte === LF) {\n      this._results.command = this._consumeTokenAsUTF8();\n      this._onByte = this._collectHeaders;\n      return;\n    }\n\n    this._consumeByte(byte);\n  }\n\n  private _collectHeaders(byte: number): void {\n    if (byte === CR) {\n      // Ignore CR\n      return;\n    }\n    if (byte === LF) {\n      this._setupCollectBody();\n      return;\n    }\n    this._onByte = this._collectHeaderKey;\n    this._reinjectByte(byte);\n  }\n\n  private _reinjectByte(byte: number) {\n    this._onByte(byte);\n  }\n\n  private _collectHeaderKey(byte: number): void {\n    if (byte === COLON) {\n      this._headerKey = this._consumeTokenAsUTF8();\n      this._onByte = this._collectHeaderValue;\n      return;\n    }\n    this._consumeByte(byte);\n  }\n\n  private _collectHeaderValue(byte: number): void {\n    if (byte === CR) {\n      // Ignore CR\n      return;\n    }\n    if (byte === LF) {\n      this._results.headers.push([\n        this._headerKey as string,\n        this._consumeTokenAsUTF8(),\n      ]);\n      this._headerKey = undefined;\n      this._onByte = this._collectHeaders;\n      return;\n    }\n    this._consumeByte(byte);\n  }\n\n  private _setupCollectBody() {\n    const contentLengthHeader = this._results.headers.filter(\n      (header: [string, string]) => {\n        return header[0] === 'content-length';\n      }\n    )[0];\n\n    if (contentLengthHeader) {\n      this._bodyBytesRemaining = parseInt(contentLengthHeader[1], 10);\n      this._onByte = this._collectBodyFixedSize;\n    } else {\n      this._onByte = this._collectBodyNullTerminated;\n    }\n  }\n\n  private _collectBodyNullTerminated(byte: number): void {\n    if (byte === NULL) {\n      this._retrievedBody();\n      return;\n    }\n    this._consumeByte(byte);\n  }\n\n  private _collectBodyFixedSize(byte: number): void {\n    // It is post decrement, so that we discard the trailing NULL octet\n    if ((this._bodyBytesRemaining as number)-- === 0) {\n      this._retrievedBody();\n      return;\n    }\n    this._consumeByte(byte);\n  }\n\n  private _retrievedBody() {\n    this._results.binaryBody = this._consumeTokenAsRaw();\n\n    try {\n      this.onFrame(this._results);\n    } catch (e) {\n      console.log(\n        `Ignoring an exception thrown by a frame handler. Original exception: `,\n        e\n      );\n    }\n\n    this._initState();\n  }\n\n  // Rec Descent Parser helpers\n\n  private _consumeByte(byte: number) {\n    this._token.push(byte);\n  }\n\n  private _consumeTokenAsUTF8() {\n    return this._decoder.decode(this._consumeTokenAsRaw());\n  }\n\n  private _consumeTokenAsRaw() {\n    const rawResult = new Uint8Array(this._token);\n    this._token = [];\n    return rawResult;\n  }\n\n  private _initState() {\n    this._results = {\n      command: undefined,\n      headers: [],\n      binaryBody: undefined,\n    };\n\n    this._token = [];\n    this._headerKey = undefined;\n\n    this._onByte = this._collectFrame;\n  }\n}\n","import { StompHeaders } from './stomp-headers';\nimport {\n  ActivationState,\n  closeEventCallbackType,\n  debugFnType,\n  frameCallbackType,\n  messageCallbackType,\n  wsErrorCallbackType,\n} from './types';\nimport { Versions } from './versions';\n\n/**\n * Configuration options for STOMP Client, each key corresponds to\n * field by the same name in {@link Client}. This can be passed to\n * the constructor of {@link Client} or to [Client#configure]{@link Client#configure}.\n *\n * There used to be a class with the same name in `@stomp/ng2-stompjs`, which has been replaced by\n * {@link RxStompConfig} and {@link InjectableRxStompConfig}.\n *\n * Part of `@stomp/stompjs`.\n */\nexport class StompConfig {\n  /**\n   * See [Client#brokerURL]{@link Client#brokerURL}.\n   */\n  public brokerURL?: string;\n\n  /**\n   * See See [Client#stompVersions]{@link Client#stompVersions}.\n   */\n  public stompVersions?: Versions;\n\n  /**\n   * See [Client#webSocketFactory]{@link Client#webSocketFactory}.\n   */\n  public webSocketFactory?: () => any;\n\n  /**\n   * See [Client#connectionTimeout]{@link Client#connectionTimeout}.\n   */\n  public connectionTimeout?: number;\n\n  /**\n   * See [Client#reconnectDelay]{@link Client#reconnectDelay}.\n   */\n  public reconnectDelay?: number;\n\n  /**\n   * See [Client#heartbeatIncoming]{@link Client#heartbeatIncoming}.\n   */\n  public heartbeatIncoming?: number;\n\n  /**\n   * See [Client#heartbeatOutgoing]{@link Client#heartbeatOutgoing}.\n   */\n  public heartbeatOutgoing?: number;\n\n  /**\n   * See [Client#splitLargeFrames]{@link Client#splitLargeFrames}.\n   */\n  public splitLargeFrames?: boolean;\n\n  /**\n   * See [Client#forceBinaryWSFrames]{@link Client#forceBinaryWSFrames}.\n   */\n  public forceBinaryWSFrames?: boolean;\n\n  /**\n   * See [Client#appendMissingNULLonIncoming]{@link Client#appendMissingNULLonIncoming}.\n   */\n  public appendMissingNULLonIncoming?: boolean;\n\n  /**\n   * See [Client#maxWebSocketChunkSize]{@link Client#maxWebSocketChunkSize}.\n   */\n  public maxWebSocketChunkSize?: number;\n\n  /**\n   * See [Client#connectHeaders]{@link Client#connectHeaders}.\n   */\n  public connectHeaders?: StompHeaders;\n\n  /**\n   * See [Client#disconnectHeaders]{@link Client#disconnectHeaders}.\n   */\n  public disconnectHeaders?: StompHeaders;\n\n  /**\n   * See [Client#onUnhandledMessage]{@link Client#onUnhandledMessage}.\n   */\n  public onUnhandledMessage?: messageCallbackType;\n\n  /**\n   * See [Client#onUnhandledReceipt]{@link Client#onUnhandledReceipt}.\n   */\n  public onUnhandledReceipt?: frameCallbackType;\n\n  /**\n   * See [Client#onUnhandledFrame]{@link Client#onUnhandledFrame}.\n   */\n  public onUnhandledFrame?: frameCallbackType;\n\n  /**\n   * See [Client#beforeConnect]{@link Client#beforeConnect}.\n   */\n  public beforeConnect?: () => void | Promise<void>;\n\n  /**\n   * See [Client#onConnect]{@link Client#onConnect}.\n   */\n  public onConnect?: frameCallbackType;\n\n  /**\n   * See [Client#onDisconnect]{@link Client#onDisconnect}.\n   */\n  public onDisconnect?: frameCallbackType;\n\n  /**\n   * See [Client#onStompError]{@link Client#onStompError}.\n   */\n  public onStompError?: frameCallbackType;\n\n  /**\n   * See [Client#onWebSocketClose]{@link Client#onWebSocketClose}.\n   */\n  public onWebSocketClose?: closeEventCallbackType;\n\n  /**\n   * See [Client#onWebSocketError]{@link Client#onWebSocketError}.\n   */\n  public onWebSocketError?: wsErrorCallbackType;\n\n  /**\n   * See [Client#logRawCommunication]{@link Client#logRawCommunication}.\n   */\n  public logRawCommunication?: boolean;\n\n  /**\n   * See [Client#debug]{@link Client#debug}.\n   */\n  public debug?: debugFnType;\n\n  /**\n   * See [Client#discardWebsocketOnCommFailure]{@link Client#discardWebsocketOnCommFailure}.\n   */\n  public discardWebsocketOnCommFailure?: boolean;\n\n  /**\n   * See [Client#onChangeState]{@link Client#onChangeState}.\n   */\n  public onChangeState?: (state: ActivationState) => void;\n}\n","import { BYTE } from './byte';\nimport { Client } from './client';\nimport { FrameImpl } from './frame-impl';\nimport { IMessage } from './i-message';\nimport { ITransaction } from './i-transaction';\nimport { Parser } from './parser';\nimport { StompHeaders } from './stomp-headers';\nimport { StompSubscription } from './stomp-subscription';\nimport {\n  closeEventCallbackType,\n  debugFnType,\n  frameCallbackType,\n  IPublishParams,\n  IStompSocket,\n  IStompSocketMessageEvent,\n  IStomptHandlerConfig,\n  messageCallbackType,\n  StompSocketState,\n  wsErrorCallbackType,\n} from './types';\nimport { Versions } from './versions';\nimport { augmentWebsocket } from './augment-websocket';\n\n/**\n * The STOMP protocol handler\n *\n * Part of `@stomp/stompjs`.\n *\n * @internal\n */\nexport class StompHandler {\n  public debug: debugFnType;\n\n  public stompVersions: Versions;\n\n  public connectHeaders: StompHeaders;\n\n  public disconnectHeaders: StompHeaders;\n\n  public heartbeatIncoming: number;\n\n  public heartbeatOutgoing: number;\n\n  public onUnhandledMessage: messageCallbackType;\n\n  public onUnhandledReceipt: frameCallbackType;\n\n  public onUnhandledFrame: frameCallbackType;\n\n  public onConnect: frameCallbackType;\n\n  public onDisconnect: frameCallbackType;\n\n  public onStompError: frameCallbackType;\n\n  public onWebSocketClose: closeEventCallbackType;\n\n  public onWebSocketError: wsErrorCallbackType;\n\n  public logRawCommunication: boolean;\n\n  public splitLargeFrames: boolean;\n\n  public maxWebSocketChunkSize: number;\n\n  public forceBinaryWSFrames: boolean;\n\n  public appendMissingNULLonIncoming: boolean;\n\n  public discardWebsocketOnCommFailure: boolean;\n\n  get connectedVersion(): string | undefined {\n    return this._connectedVersion;\n  }\n  private _connectedVersion: string | undefined;\n\n  get connected(): boolean {\n    return this._connected;\n  }\n\n  private _connected: boolean = false;\n\n  private readonly _subscriptions: { [key: string]: messageCallbackType };\n  private readonly _receiptWatchers: { [key: string]: frameCallbackType };\n  private _partialData: string;\n  private _escapeHeaderValues: boolean;\n  private _counter: number;\n  private _pinger: any;\n  private _ponger: any;\n  private _lastServerActivityTS: number;\n\n  constructor(\n    private _client: Client,\n    public _webSocket: IStompSocket,\n    config: IStomptHandlerConfig\n  ) {\n    // used to index subscribers\n    this._counter = 0;\n\n    // subscription callbacks indexed by subscriber's ID\n    this._subscriptions = {};\n\n    // receipt-watchers indexed by receipts-ids\n    this._receiptWatchers = {};\n\n    this._partialData = '';\n\n    this._escapeHeaderValues = false;\n\n    this._lastServerActivityTS = Date.now();\n\n    this.debug = config.debug;\n    this.stompVersions = config.stompVersions;\n    this.connectHeaders = config.connectHeaders;\n    this.disconnectHeaders = config.disconnectHeaders;\n    this.heartbeatIncoming = config.heartbeatIncoming;\n    this.heartbeatOutgoing = config.heartbeatOutgoing;\n    this.splitLargeFrames = config.splitLargeFrames;\n    this.maxWebSocketChunkSize = config.maxWebSocketChunkSize;\n    this.forceBinaryWSFrames = config.forceBinaryWSFrames;\n    this.logRawCommunication = config.logRawCommunication;\n    this.appendMissingNULLonIncoming = config.appendMissingNULLonIncoming;\n    this.discardWebsocketOnCommFailure = config.discardWebsocketOnCommFailure;\n    this.onConnect = config.onConnect;\n    this.onDisconnect = config.onDisconnect;\n    this.onStompError = config.onStompError;\n    this.onWebSocketClose = config.onWebSocketClose;\n    this.onWebSocketError = config.onWebSocketError;\n    this.onUnhandledMessage = config.onUnhandledMessage;\n    this.onUnhandledReceipt = config.onUnhandledReceipt;\n    this.onUnhandledFrame = config.onUnhandledFrame;\n  }\n\n  public start(): void {\n    const parser = new Parser(\n      // On Frame\n      rawFrame => {\n        const frame = FrameImpl.fromRawFrame(\n          rawFrame,\n          this._escapeHeaderValues\n        );\n\n        // if this.logRawCommunication is set, the rawChunk is logged at this._webSocket.onmessage\n        if (!this.logRawCommunication) {\n          this.debug(`<<< ${frame}`);\n        }\n\n        const serverFrameHandler =\n          this._serverFrameHandlers[frame.command] || this.onUnhandledFrame;\n        serverFrameHandler(frame);\n      },\n      // On Incoming Ping\n      () => {\n        this.debug('<<< PONG');\n      }\n    );\n\n    this._webSocket.onmessage = (evt: IStompSocketMessageEvent) => {\n      this.debug('Received data');\n      this._lastServerActivityTS = Date.now();\n\n      if (this.logRawCommunication) {\n        const rawChunkAsString =\n          evt.data instanceof ArrayBuffer\n            ? new TextDecoder().decode(evt.data)\n            : evt.data;\n        this.debug(`<<< ${rawChunkAsString}`);\n      }\n\n      parser.parseChunk(\n        evt.data as string | ArrayBuffer,\n        this.appendMissingNULLonIncoming\n      );\n    };\n\n    this._webSocket.onclose = (closeEvent): void => {\n      this.debug(`Connection closed to ${this._client.brokerURL}`);\n      this._cleanUp();\n      this.onWebSocketClose(closeEvent);\n    };\n\n    this._webSocket.onerror = (errorEvent): void => {\n      this.onWebSocketError(errorEvent);\n    };\n\n    this._webSocket.onopen = () => {\n      // Clone before updating\n      const connectHeaders = (Object as any).assign({}, this.connectHeaders);\n\n      this.debug('Web Socket Opened...');\n      connectHeaders['accept-version'] = this.stompVersions.supportedVersions();\n      connectHeaders['heart-beat'] = [\n        this.heartbeatOutgoing,\n        this.heartbeatIncoming,\n      ].join(',');\n      this._transmit({ command: 'CONNECT', headers: connectHeaders });\n    };\n  }\n\n  private readonly _serverFrameHandlers: {\n    [key: string]: frameCallbackType;\n  } = {\n    // [CONNECTED Frame](http://stomp.github.com/stomp-specification-1.2.html#CONNECTED_Frame)\n    CONNECTED: frame => {\n      this.debug(`connected to server ${frame.headers.server}`);\n      this._connected = true;\n      this._connectedVersion = frame.headers.version;\n      // STOMP version 1.2 needs header values to be escaped\n      if (this._connectedVersion === Versions.V1_2) {\n        this._escapeHeaderValues = true;\n      }\n\n      this._setupHeartbeat(frame.headers);\n      this.onConnect(frame);\n    },\n\n    // [MESSAGE Frame](http://stomp.github.com/stomp-specification-1.2.html#MESSAGE)\n    MESSAGE: frame => {\n      // the callback is registered when the client calls\n      // `subscribe()`.\n      // If there is no registered subscription for the received message,\n      // the default `onUnhandledMessage` callback is used that the client can set.\n      // This is useful for subscriptions that are automatically created\n      // on the browser side (e.g. [RabbitMQ's temporary\n      // queues](http://www.rabbitmq.com/stomp.html)).\n      const subscription = frame.headers.subscription;\n      const onReceive =\n        this._subscriptions[subscription] || this.onUnhandledMessage;\n\n      // bless the frame to be a Message\n      const message = frame as IMessage;\n\n      const client = this;\n      const messageId =\n        this._connectedVersion === Versions.V1_2\n          ? message.headers.ack\n          : message.headers['message-id'];\n\n      // add `ack()` and `nack()` methods directly to the returned frame\n      // so that a simple call to `message.ack()` can acknowledge the message.\n      message.ack = (headers: StompHeaders = {}): void => {\n        return client.ack(messageId, subscription, headers);\n      };\n      message.nack = (headers: StompHeaders = {}): void => {\n        return client.nack(messageId, subscription, headers);\n      };\n      onReceive(message);\n    },\n\n    // [RECEIPT Frame](http://stomp.github.com/stomp-specification-1.2.html#RECEIPT)\n    RECEIPT: frame => {\n      const callback = this._receiptWatchers[frame.headers['receipt-id']];\n      if (callback) {\n        callback(frame);\n        // Server will acknowledge only once, remove the callback\n        delete this._receiptWatchers[frame.headers['receipt-id']];\n      } else {\n        this.onUnhandledReceipt(frame);\n      }\n    },\n\n    // [ERROR Frame](http://stomp.github.com/stomp-specification-1.2.html#ERROR)\n    ERROR: frame => {\n      this.onStompError(frame);\n    },\n  };\n\n  private _setupHeartbeat(headers: StompHeaders): void {\n    if (\n      headers.version !== Versions.V1_1 &&\n      headers.version !== Versions.V1_2\n    ) {\n      return;\n    }\n\n    // It is valid for the server to not send this header\n    // https://stomp.github.io/stomp-specification-1.2.html#Heart-beating\n    if (!headers['heart-beat']) {\n      return;\n    }\n\n    // heart-beat header received from the server looks like:\n    //\n    //     heart-beat: sx, sy\n    const [serverOutgoing, serverIncoming] = headers['heart-beat']\n      .split(',')\n      .map((v: string) => parseInt(v, 10));\n\n    if (this.heartbeatOutgoing !== 0 && serverIncoming !== 0) {\n      const ttl: number = Math.max(this.heartbeatOutgoing, serverIncoming);\n      this.debug(`send PING every ${ttl}ms`);\n      this._pinger = setInterval(() => {\n        if (this._webSocket.readyState === StompSocketState.OPEN) {\n          this._webSocket.send(BYTE.LF);\n          this.debug('>>> PING');\n        }\n      }, ttl);\n    }\n\n    if (this.heartbeatIncoming !== 0 && serverOutgoing !== 0) {\n      const ttl: number = Math.max(this.heartbeatIncoming, serverOutgoing);\n      this.debug(`check PONG every ${ttl}ms`);\n      this._ponger = setInterval(() => {\n        const delta = Date.now() - this._lastServerActivityTS;\n        // We wait twice the TTL to be flexible on window's setInterval calls\n        if (delta > ttl * 2) {\n          this.debug(`did not receive server activity for the last ${delta}ms`);\n          this._closeOrDiscardWebsocket();\n        }\n      }, ttl);\n    }\n  }\n\n  private _closeOrDiscardWebsocket() {\n    if (this.discardWebsocketOnCommFailure) {\n      this.debug(\n        'Discarding websocket, the underlying socket may linger for a while'\n      );\n      this.discardWebsocket();\n    } else {\n      this.debug('Issuing close on the websocket');\n      this._closeWebsocket();\n    }\n  }\n\n  public forceDisconnect() {\n    if (this._webSocket) {\n      if (\n        this._webSocket.readyState === StompSocketState.CONNECTING ||\n        this._webSocket.readyState === StompSocketState.OPEN\n      ) {\n        this._closeOrDiscardWebsocket();\n      }\n    }\n  }\n\n  public _closeWebsocket() {\n    this._webSocket.onmessage = () => {}; // ignore messages\n    this._webSocket.close();\n  }\n\n  public discardWebsocket() {\n    if (typeof this._webSocket.terminate !== 'function') {\n      augmentWebsocket(this._webSocket, (msg: string) => this.debug(msg));\n    }\n\n    // @ts-ignore - this method will be there at this stage\n    this._webSocket.terminate();\n  }\n\n  private _transmit(params: {\n    command: string;\n    headers?: StompHeaders;\n    body?: string;\n    binaryBody?: Uint8Array;\n    skipContentLengthHeader?: boolean;\n  }): void {\n    const { command, headers, body, binaryBody, skipContentLengthHeader } =\n      params;\n    const frame = new FrameImpl({\n      command,\n      headers,\n      body,\n      binaryBody,\n      escapeHeaderValues: this._escapeHeaderValues,\n      skipContentLengthHeader,\n    });\n\n    let rawChunk = frame.serialize();\n\n    if (this.logRawCommunication) {\n      this.debug(`>>> ${rawChunk}`);\n    } else {\n      this.debug(`>>> ${frame}`);\n    }\n\n    if (this.forceBinaryWSFrames && typeof rawChunk === 'string') {\n      rawChunk = new TextEncoder().encode(rawChunk);\n    }\n\n    if (typeof rawChunk !== 'string' || !this.splitLargeFrames) {\n      this._webSocket.send(rawChunk);\n    } else {\n      let out = rawChunk as string;\n      while (out.length > 0) {\n        const chunk = out.substring(0, this.maxWebSocketChunkSize);\n        out = out.substring(this.maxWebSocketChunkSize);\n        this._webSocket.send(chunk);\n        this.debug(`chunk sent = ${chunk.length}, remaining = ${out.length}`);\n      }\n    }\n  }\n\n  public dispose(): void {\n    if (this.connected) {\n      try {\n        // clone before updating\n        const disconnectHeaders = (Object as any).assign(\n          {},\n          this.disconnectHeaders\n        );\n\n        if (!disconnectHeaders.receipt) {\n          disconnectHeaders.receipt = `close-${this._counter++}`;\n        }\n        this.watchForReceipt(disconnectHeaders.receipt, frame => {\n          this._closeWebsocket();\n          this._cleanUp();\n          this.onDisconnect(frame);\n        });\n        this._transmit({ command: 'DISCONNECT', headers: disconnectHeaders });\n      } catch (error) {\n        this.debug(`Ignoring error during disconnect ${error}`);\n      }\n    } else {\n      if (\n        this._webSocket.readyState === StompSocketState.CONNECTING ||\n        this._webSocket.readyState === StompSocketState.OPEN\n      ) {\n        this._closeWebsocket();\n      }\n    }\n  }\n\n  private _cleanUp() {\n    this._connected = false;\n\n    if (this._pinger) {\n      clearInterval(this._pinger);\n      this._pinger = undefined;\n    }\n    if (this._ponger) {\n      clearInterval(this._ponger);\n      this._ponger = undefined;\n    }\n  }\n\n  public publish(params: IPublishParams): void {\n    const { destination, headers, body, binaryBody, skipContentLengthHeader } =\n      params;\n    const hdrs: StompHeaders = (Object as any).assign({ destination }, headers);\n    this._transmit({\n      command: 'SEND',\n      headers: hdrs,\n      body,\n      binaryBody,\n      skipContentLengthHeader,\n    });\n  }\n\n  public watchForReceipt(receiptId: string, callback: frameCallbackType): void {\n    this._receiptWatchers[receiptId] = callback;\n  }\n\n  public subscribe(\n    destination: string,\n    callback: messageCallbackType,\n    headers: StompHeaders = {}\n  ): StompSubscription {\n    headers = (Object as any).assign({}, headers);\n\n    if (!headers.id) {\n      headers.id = `sub-${this._counter++}`;\n    }\n    headers.destination = destination;\n    this._subscriptions[headers.id] = callback;\n    this._transmit({ command: 'SUBSCRIBE', headers });\n    const client = this;\n    return {\n      id: headers.id,\n\n      unsubscribe(hdrs) {\n        return client.unsubscribe(headers.id, hdrs);\n      },\n    };\n  }\n\n  public unsubscribe(id: string, headers: StompHeaders = {}): void {\n    headers = (Object as any).assign({}, headers);\n\n    delete this._subscriptions[id];\n    headers.id = id;\n    this._transmit({ command: 'UNSUBSCRIBE', headers });\n  }\n\n  public begin(transactionId: string): ITransaction {\n    const txId = transactionId || `tx-${this._counter++}`;\n    this._transmit({\n      command: 'BEGIN',\n      headers: {\n        transaction: txId,\n      },\n    });\n    const client = this;\n    return {\n      id: txId,\n      commit(): void {\n        client.commit(txId);\n      },\n      abort(): void {\n        client.abort(txId);\n      },\n    };\n  }\n\n  public commit(transactionId: string): void {\n    this._transmit({\n      command: 'COMMIT',\n      headers: {\n        transaction: transactionId,\n      },\n    });\n  }\n\n  public abort(transactionId: string): void {\n    this._transmit({\n      command: 'ABORT',\n      headers: {\n        transaction: transactionId,\n      },\n    });\n  }\n\n  public ack(\n    messageId: string,\n    subscriptionId: string,\n    headers: StompHeaders = {}\n  ): void {\n    headers = (Object as any).assign({}, headers);\n\n    if (this._connectedVersion === Versions.V1_2) {\n      headers.id = messageId;\n    } else {\n      headers['message-id'] = messageId;\n    }\n    headers.subscription = subscriptionId;\n    this._transmit({ command: 'ACK', headers });\n  }\n\n  public nack(\n    messageId: string,\n    subscriptionId: string,\n    headers: StompHeaders = {}\n  ): void {\n    headers = (Object as any).assign({}, headers);\n\n    if (this._connectedVersion === Versions.V1_2) {\n      headers.id = messageId;\n    } else {\n      headers['message-id'] = messageId;\n    }\n    headers.subscription = subscriptionId;\n    return this._transmit({ command: 'NACK', headers });\n  }\n}\n","/**\n * STOMP headers. Many functions calls will accept headers as parameters.\n * The headers sent by Broker will be available as [IFrame#headers]{@link IFrame#headers}.\n *\n * `key` and `value` must be valid strings.\n * In addition, `key` must not contain `CR`, `LF`, or `:`.\n *\n * Part of `@stomp/stompjs`.\n */\nexport class StompHeaders {\n  [key: string]: string;\n}\n","import { IFrame } from './i-frame';\nimport { IMessage } from './i-message';\nimport { StompHeaders } from './stomp-headers';\nimport { Versions } from './versions';\n\n/**\n * This callback will receive a `string` as parameter.\n *\n * Part of `@stomp/stompjs`.\n */\nexport type debugFnType = (msg: string) => void;\n\n/**\n * This callback will receive a {@link IMessage} as parameter.\n *\n * Part of `@stomp/stompjs`.\n */\nexport type messageCallbackType = (message: IMessage) => void;\n\n/**\n * This callback will receive a {@link IFrame} as parameter.\n *\n * Part of `@stomp/stompjs`.\n */\nexport type frameCallbackType = (receipt: IFrame) => void;\n\n/**\n * This callback will receive a [CloseEvent]{@link https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent}\n * as parameter.\n *\n * Part of `@stomp/stompjs`.\n */\nexport type closeEventCallbackType<T = any> = (evt: T) => void;\n\n/**\n * This callback will receive an [Event]{@link https://developer.mozilla.org/en-US/docs/Web/API/Event}\n * as parameter.\n *\n * Part of `@stomp/stompjs`.\n */\nexport type wsErrorCallbackType<T = any> = (evt: T) => void;\n\n/**\n * Parameters for [Client#publish]{@link Client#publish}.\n * Aliased as publishParams as well.\n *\n * Part of `@stomp/stompjs`.\n */\nexport interface IPublishParams {\n  /**\n   * destination end point\n   */\n  destination: string;\n  /**\n   * headers (optional)\n   */\n  headers?: StompHeaders;\n  /**\n   * body (optional)\n   */\n  body?: string;\n  /**\n   * binary body (optional)\n   */\n  binaryBody?: Uint8Array;\n  /**\n   * By default a `content-length` header will be added in the Frame to the broker.\n   * Set it to `true` for the header to be skipped.\n   */\n  skipContentLengthHeader?: boolean;\n}\n\n/**\n * Backward compatibility, switch to {@link IPublishParams}.\n */\nexport type publishParams = IPublishParams;\n\n/**\n * Used in {@link IRawFrameType}\n *\n * Part of `@stomp/stompjs`.\n *\n * @internal\n */\nexport type RawHeaderType = [string, string];\n\n/**\n * The parser yield frames in this structure\n *\n * Part of `@stomp/stompjs`.\n *\n * @internal\n */\nexport interface IRawFrameType {\n  command: string | undefined;\n  headers: RawHeaderType[];\n  binaryBody: Uint8Array | undefined;\n}\n\n/**\n * @internal\n */\nexport interface IStompSocketMessageEvent {\n  data?: string | ArrayBuffer;\n}\n\n/**\n * Copied from Websocket interface to avoid dom typelib dependency.\n *\n * @internal\n */\nexport interface IStompSocket {\n  onclose: ((this: IStompSocket, ev?: any) => any) | null;\n  onerror: ((this: IStompSocket, ev: any) => any) | null;\n  onmessage: ((this: IStompSocket, ev: IStompSocketMessageEvent) => any) | null;\n  onopen: ((this: IStompSocket, ev?: any) => any) | null;\n  terminate?: ((this: IStompSocket) => any) | null;\n\n  /**\n   * Returns a string that indicates how binary data from the socket is exposed to scripts:\n   * We support only 'arraybuffer'.\n   */\n  binaryType: 'arraybuffer';\n\n  /**\n   * Returns the state of the socket connection. It can have the values of StompSocketState.\n   */\n  readonly readyState: number;\n\n  /**\n   * Closes the connection.\n   */\n  close(): void;\n  /**\n   * Transmits data using the connection. data can be a string or an ArrayBuffer.\n   */\n  send(data: string | ArrayBuffer): void;\n}\n\n/**\n * Possible states for the IStompSocket\n */\nexport enum StompSocketState {\n  CONNECTING,\n  OPEN,\n  CLOSING,\n  CLOSED,\n}\n\n/**\n * Possible activation state\n */\nexport enum ActivationState {\n  ACTIVE,\n  DEACTIVATING,\n  INACTIVE,\n}\n\n/**\n * @internal\n */\nexport interface IStomptHandlerConfig {\n  debug: debugFnType;\n  stompVersions: Versions;\n  connectHeaders: StompHeaders;\n  disconnectHeaders: StompHeaders;\n  heartbeatIncoming: number;\n  heartbeatOutgoing: number;\n  splitLargeFrames: boolean;\n  maxWebSocketChunkSize: number;\n  forceBinaryWSFrames: boolean;\n  logRawCommunication: boolean;\n  appendMissingNULLonIncoming: boolean;\n  discardWebsocketOnCommFailure: boolean;\n  onConnect: frameCallbackType;\n  onDisconnect: frameCallbackType;\n  onStompError: frameCallbackType;\n  onWebSocketClose: closeEventCallbackType;\n  onWebSocketError: wsErrorCallbackType;\n  onUnhandledMessage: messageCallbackType;\n  onUnhandledReceipt: frameCallbackType;\n  onUnhandledFrame: frameCallbackType;\n}\n","/**\n * Supported STOMP versions\n *\n * Part of `@stomp/stompjs`.\n */\nexport class Versions {\n  /**\n   * Indicates protocol version 1.0\n   */\n  public static V1_0 = '1.0';\n  /**\n   * Indicates protocol version 1.1\n   */\n  public static V1_1 = '1.1';\n  /**\n   * Indicates protocol version 1.2\n   */\n  public static V1_2 = '1.2';\n\n  /**\n   * @internal\n   */\n  public static default = new Versions([\n    Versions.V1_2,\n    Versions.V1_1,\n    Versions.V1_0,\n  ]);\n\n  /**\n   * Takes an array of string of versions, typical elements '1.0', '1.1', or '1.2'\n   *\n   * You will an instance if this class if you want to override supported versions to be declared during\n   * STOMP handshake.\n   */\n  constructor(public versions: string[]) {}\n\n  /**\n   * Used as part of CONNECT STOMP Frame\n   */\n  public supportedVersions() {\n    return this.versions.join(',');\n  }\n\n  /**\n   * Used while creating a WebSocket\n   */\n  public protocolVersions() {\n    return this.versions.map(x => `v${x.replace('.', '')}.stomp`);\n  }\n}\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","export * from './client';\nexport * from './frame-impl';\nexport * from './i-frame';\nexport * from './i-message';\nexport * from './parser';\nexport * from './stomp-config';\nexport * from './stomp-headers';\nexport * from './stomp-subscription';\nexport * from './i-transaction';\nexport * from './types';\nexport * from './versions';\n\n// Compatibility code\nexport * from './compatibility/compat-client';\nexport * from './compatibility/stomp';\n"],"names":[],"sourceRoot":""}
1858
+ // tslint:disable-next-line:variable-name
1859
+ Stomp.WebSocketClass = null;
1860
+
1861
+ exports.Client = Client;
1862
+ exports.CompatClient = CompatClient;
1863
+ exports.FrameImpl = FrameImpl;
1864
+ exports.Parser = Parser;
1865
+ exports.Stomp = Stomp;
1866
+ exports.StompConfig = StompConfig;
1867
+ exports.StompHeaders = StompHeaders;
1868
+ exports.Versions = Versions;
1869
+
1870
+ }));
1871
+ //# sourceMappingURL=stomp.umd.js.map