@robdobsn/raftjs 1.8.5 → 1.10.7

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 (224) hide show
  1. package/.editorconfig +14 -0
  2. package/.gitattributes +11 -0
  3. package/.nvmrc +1 -0
  4. package/TODO.md +1 -0
  5. package/dist/react-native/RaftAttributeHandler.d.ts +14 -0
  6. package/dist/react-native/RaftAttributeHandler.js +375 -0
  7. package/dist/react-native/RaftAttributeHandler.js.map +1 -0
  8. package/dist/react-native/RaftChannel.d.ts +20 -0
  9. package/dist/react-native/RaftChannel.js +12 -0
  10. package/dist/react-native/RaftChannel.js.map +1 -0
  11. package/dist/react-native/RaftChannelBLE.native.d.ts +95 -0
  12. package/dist/react-native/RaftChannelBLE.native.js +483 -0
  13. package/dist/react-native/RaftChannelBLE.native.js.map +1 -0
  14. package/dist/react-native/RaftChannelBLE.web.d.ts +40 -0
  15. package/dist/react-native/RaftChannelBLE.web.js +302 -0
  16. package/dist/react-native/RaftChannelBLE.web.js.map +1 -0
  17. package/dist/react-native/RaftChannelBLEFactory.d.ts +10 -0
  18. package/dist/react-native/RaftChannelBLEFactory.js +17 -0
  19. package/dist/react-native/RaftChannelBLEFactory.js.map +1 -0
  20. package/dist/react-native/RaftChannelBLEScanner.native.d.ts +18 -0
  21. package/dist/react-native/RaftChannelBLEScanner.native.js +138 -0
  22. package/dist/react-native/RaftChannelBLEScanner.native.js.map +1 -0
  23. package/dist/react-native/RaftChannelSimulated.d.ts +42 -0
  24. package/dist/react-native/RaftChannelSimulated.js +1000 -0
  25. package/dist/react-native/RaftChannelSimulated.js.map +1 -0
  26. package/dist/react-native/RaftChannelWebSerial.d.ts +39 -0
  27. package/dist/react-native/RaftChannelWebSerial.js +329 -0
  28. package/dist/react-native/RaftChannelWebSerial.js.map +1 -0
  29. package/dist/react-native/RaftChannelWebSocket.d.ts +30 -0
  30. package/dist/react-native/RaftChannelWebSocket.js +222 -0
  31. package/dist/react-native/RaftChannelWebSocket.js.map +1 -0
  32. package/dist/react-native/RaftCommsStats.d.ts +39 -0
  33. package/dist/react-native/RaftCommsStats.js +128 -0
  34. package/dist/react-native/RaftCommsStats.js.map +1 -0
  35. package/dist/react-native/RaftConnEvents.d.ts +39 -0
  36. package/dist/react-native/RaftConnEvents.js +54 -0
  37. package/dist/react-native/RaftConnEvents.js.map +1 -0
  38. package/dist/react-native/RaftConnector.d.ts +248 -0
  39. package/dist/react-native/RaftConnector.js +658 -0
  40. package/dist/react-native/RaftConnector.js.map +1 -0
  41. package/dist/react-native/RaftCustomAttrHandler.d.ts +6 -0
  42. package/dist/react-native/RaftCustomAttrHandler.js +93 -0
  43. package/dist/react-native/RaftCustomAttrHandler.js.map +1 -0
  44. package/dist/react-native/RaftDeviceInfo.d.ts +71 -0
  45. package/dist/react-native/RaftDeviceInfo.js +50 -0
  46. package/dist/react-native/RaftDeviceInfo.js.map +1 -0
  47. package/dist/react-native/RaftDeviceManager.d.ts +61 -0
  48. package/dist/react-native/RaftDeviceManager.js +665 -0
  49. package/dist/react-native/RaftDeviceManager.js.map +1 -0
  50. package/dist/react-native/RaftDeviceMgrIF.d.ts +15 -0
  51. package/dist/react-native/RaftDeviceMgrIF.js +11 -0
  52. package/dist/react-native/RaftDeviceMgrIF.js.map +1 -0
  53. package/dist/react-native/RaftDeviceMsg.d.ts +9 -0
  54. package/dist/react-native/RaftDeviceMsg.js +11 -0
  55. package/dist/react-native/RaftDeviceMsg.js.map +1 -0
  56. package/dist/react-native/RaftDeviceStates.d.ts +37 -0
  57. package/dist/react-native/RaftDeviceStates.js +60 -0
  58. package/dist/react-native/RaftDeviceStates.js.map +1 -0
  59. package/dist/react-native/RaftFileHandler.d.ts +52 -0
  60. package/dist/react-native/RaftFileHandler.js +502 -0
  61. package/dist/react-native/RaftFileHandler.js.map +1 -0
  62. package/dist/react-native/RaftLog.d.ts +22 -0
  63. package/dist/react-native/RaftLog.js +63 -0
  64. package/dist/react-native/RaftLog.js.map +1 -0
  65. package/dist/react-native/RaftMiniHDLC.d.ts +18 -0
  66. package/dist/react-native/RaftMiniHDLC.js +383 -0
  67. package/dist/react-native/RaftMiniHDLC.js.map +1 -0
  68. package/dist/react-native/RaftMsgHandler.d.ts +62 -0
  69. package/dist/react-native/RaftMsgHandler.js +511 -0
  70. package/dist/react-native/RaftMsgHandler.js.map +1 -0
  71. package/dist/react-native/RaftMsgTrackInfo.d.ts +17 -0
  72. package/dist/react-native/RaftMsgTrackInfo.js +42 -0
  73. package/dist/react-native/RaftMsgTrackInfo.js.map +1 -0
  74. package/dist/react-native/RaftProtocolDefs.d.ts +30 -0
  75. package/dist/react-native/RaftProtocolDefs.js +48 -0
  76. package/dist/react-native/RaftProtocolDefs.js.map +1 -0
  77. package/dist/react-native/RaftStreamHandler.d.ts +38 -0
  78. package/dist/react-native/RaftStreamHandler.js +258 -0
  79. package/dist/react-native/RaftStreamHandler.js.map +1 -0
  80. package/dist/react-native/RaftStruct.d.ts +3 -0
  81. package/dist/react-native/RaftStruct.js +258 -0
  82. package/dist/react-native/RaftStruct.js.map +1 -0
  83. package/dist/react-native/RaftSysTypeManager.d.ts +16 -0
  84. package/dist/react-native/RaftSysTypeManager.js +78 -0
  85. package/dist/react-native/RaftSysTypeManager.js.map +1 -0
  86. package/dist/react-native/RaftSystemType.d.ts +30 -0
  87. package/dist/react-native/RaftSystemType.js +3 -0
  88. package/dist/react-native/RaftSystemType.js.map +1 -0
  89. package/dist/react-native/RaftSystemUtils.d.ts +136 -0
  90. package/dist/react-native/RaftSystemUtils.js +412 -0
  91. package/dist/react-native/RaftSystemUtils.js.map +1 -0
  92. package/dist/react-native/RaftTypes.d.ts +195 -0
  93. package/dist/react-native/RaftTypes.js +153 -0
  94. package/dist/react-native/RaftTypes.js.map +1 -0
  95. package/dist/react-native/RaftUpdateEvents.d.ts +33 -0
  96. package/dist/react-native/RaftUpdateEvents.js +46 -0
  97. package/dist/react-native/RaftUpdateEvents.js.map +1 -0
  98. package/dist/react-native/RaftUpdateManager.d.ts +61 -0
  99. package/dist/react-native/RaftUpdateManager.js +621 -0
  100. package/dist/react-native/RaftUpdateManager.js.map +1 -0
  101. package/dist/react-native/RaftUtils.d.ts +128 -0
  102. package/dist/react-native/RaftUtils.js +487 -0
  103. package/dist/react-native/RaftUtils.js.map +1 -0
  104. package/dist/react-native/RaftWifiTypes.d.ts +23 -0
  105. package/dist/react-native/RaftWifiTypes.js +43 -0
  106. package/dist/react-native/RaftWifiTypes.js.map +1 -0
  107. package/dist/react-native/main.d.ts +26 -0
  108. package/dist/react-native/main.js +51 -0
  109. package/dist/react-native/main.js.map +1 -0
  110. package/dist/web/RaftAttributeHandler.js +1 -1
  111. package/dist/web/RaftAttributeHandler.js.map +1 -1
  112. package/dist/web/RaftChannelBLE.web.js +8 -6
  113. package/dist/web/RaftChannelBLE.web.js.map +1 -1
  114. package/dist/web/RaftChannelSimulated.d.ts +10 -0
  115. package/dist/web/RaftChannelSimulated.js +662 -80
  116. package/dist/web/RaftChannelSimulated.js.map +1 -1
  117. package/dist/web/RaftChannelWebSerial.js +2 -2
  118. package/dist/web/RaftChannelWebSerial.js.map +1 -1
  119. package/dist/web/RaftChannelWebSocket.js +16 -1
  120. package/dist/web/RaftChannelWebSocket.js.map +1 -1
  121. package/dist/web/RaftConnector.d.ts +2 -0
  122. package/dist/web/RaftConnector.js +38 -15
  123. package/dist/web/RaftConnector.js.map +1 -1
  124. package/dist/web/RaftCustomAttrHandler.d.ts +2 -0
  125. package/dist/web/RaftCustomAttrHandler.js +54 -26
  126. package/dist/web/RaftCustomAttrHandler.js.map +1 -1
  127. package/dist/web/RaftDeviceInfo.d.ts +3 -1
  128. package/dist/web/RaftDeviceInfo.js +17 -3
  129. package/dist/web/RaftDeviceInfo.js.map +1 -1
  130. package/dist/web/RaftDeviceManager.d.ts +19 -1
  131. package/dist/web/RaftDeviceManager.js +89 -3
  132. package/dist/web/RaftDeviceManager.js.map +1 -1
  133. package/dist/web/RaftDeviceStates.d.ts +1 -1
  134. package/dist/web/RaftDeviceStates.js +2 -2
  135. package/dist/web/RaftDeviceStates.js.map +1 -1
  136. package/dist/web/RaftMsgHandler.js.map +1 -1
  137. package/dist/web/RaftStreamHandler.js +2 -1
  138. package/dist/web/RaftStreamHandler.js.map +1 -1
  139. package/dist/web/RaftStruct.js +197 -147
  140. package/dist/web/RaftStruct.js.map +1 -1
  141. package/dist/web/RaftUpdateManager.js +1 -1
  142. package/dist/web/RaftUpdateManager.js.map +1 -1
  143. package/dist/web/RaftUtils.d.ts +2 -0
  144. package/dist/web/RaftUtils.js +20 -0
  145. package/dist/web/RaftUtils.js.map +1 -1
  146. package/dist/web/main.d.ts +1 -0
  147. package/dist/web/main.js.map +1 -1
  148. package/eslint.config.mjs +33 -0
  149. package/examples/dashboard/package.json +36 -0
  150. package/examples/dashboard/src/CommandPanel.tsx +147 -0
  151. package/examples/dashboard/src/ConnManager.ts +166 -0
  152. package/examples/dashboard/src/DeviceActionsForm.tsx +133 -0
  153. package/examples/dashboard/src/DeviceAttrsForm.tsx +49 -0
  154. package/examples/dashboard/src/DeviceLineChart.tsx +163 -0
  155. package/examples/dashboard/src/DevicePanel.tsx +171 -0
  156. package/examples/dashboard/src/DevicesPanel.tsx +58 -0
  157. package/examples/dashboard/src/DispLedGrid.tsx +110 -0
  158. package/examples/dashboard/src/DispOneLed.tsx +20 -0
  159. package/examples/dashboard/src/LatencyTest.ts +130 -0
  160. package/examples/dashboard/src/LatencyTestPanel.tsx +92 -0
  161. package/examples/dashboard/src/Main.tsx +234 -0
  162. package/examples/dashboard/src/SettingsManager.ts +67 -0
  163. package/examples/dashboard/src/SettingsScreen.tsx +174 -0
  164. package/examples/dashboard/src/StatusPanel.tsx +71 -0
  165. package/examples/dashboard/src/SystemTypeCog/CogStateInfo.ts +162 -0
  166. package/examples/dashboard/src/SystemTypeCog/SystemTypeCog.ts +91 -0
  167. package/examples/dashboard/src/SystemTypeGeneric/StateInfoGeneric.ts +30 -0
  168. package/examples/dashboard/src/SystemTypeGeneric/SystemTypeGeneric.ts +91 -0
  169. package/examples/dashboard/src/SystemTypeMarty/RICAddOn.ts +70 -0
  170. package/examples/dashboard/src/SystemTypeMarty/RICAddOnBase.ts +33 -0
  171. package/examples/dashboard/src/SystemTypeMarty/RICAddOnManager.ts +342 -0
  172. package/examples/dashboard/src/SystemTypeMarty/RICCommsStats.ts +170 -0
  173. package/examples/dashboard/src/SystemTypeMarty/RICHWElem.ts +123 -0
  174. package/examples/dashboard/src/SystemTypeMarty/RICLEDPatternChecker.ts +207 -0
  175. package/examples/dashboard/src/SystemTypeMarty/RICROSSerial.ts +464 -0
  176. package/examples/dashboard/src/SystemTypeMarty/RICServoFaultDetector.ts +146 -0
  177. package/examples/dashboard/src/SystemTypeMarty/RICStateInfo.ts +97 -0
  178. package/examples/dashboard/src/SystemTypeMarty/RICSystemUtils.ts +371 -0
  179. package/examples/dashboard/src/SystemTypeMarty/RICTypes.ts +20 -0
  180. package/examples/dashboard/src/SystemTypeMarty/SystemTypeMarty.ts +119 -0
  181. package/examples/dashboard/src/index.html +15 -0
  182. package/examples/dashboard/src/index.tsx +13 -0
  183. package/examples/dashboard/src/styles.css +408 -0
  184. package/examples/dashboard/tsconfig.json +18 -0
  185. package/jest.config.js +11 -0
  186. package/package.json +4 -7
  187. package/src/RaftAttributeHandler.ts +450 -0
  188. package/src/RaftChannel.ts +32 -0
  189. package/src/RaftChannelBLE.native.ts +617 -0
  190. package/src/RaftChannelBLE.web.ts +374 -0
  191. package/src/RaftChannelBLEFactory.ts +13 -0
  192. package/src/RaftChannelBLEScanner.native.ts +184 -0
  193. package/src/RaftChannelSimulated.ts +1176 -0
  194. package/src/RaftChannelWebSerial.ts +420 -0
  195. package/src/RaftChannelWebSocket.ts +272 -0
  196. package/src/RaftCommsStats.ts +142 -0
  197. package/src/RaftConnEvents.ts +58 -0
  198. package/src/RaftConnector.ts +785 -0
  199. package/src/RaftCustomAttrHandler.ts +117 -0
  200. package/src/RaftDeviceInfo.ts +125 -0
  201. package/src/RaftDeviceManager.ts +844 -0
  202. package/src/RaftDeviceMgrIF.ts +33 -0
  203. package/src/RaftDeviceMsg.ts +20 -0
  204. package/src/RaftDeviceStates.ts +92 -0
  205. package/src/RaftFileHandler.ts +668 -0
  206. package/src/RaftLog.ts +70 -0
  207. package/src/RaftMiniHDLC.ts +396 -0
  208. package/src/RaftMsgHandler.ts +812 -0
  209. package/src/RaftMsgTrackInfo.ts +51 -0
  210. package/src/RaftProtocolDefs.ts +46 -0
  211. package/src/RaftStreamHandler.ts +329 -0
  212. package/src/RaftStruct.ts +282 -0
  213. package/src/RaftSysTypeManager.ts +87 -0
  214. package/src/RaftSystemType.ts +34 -0
  215. package/src/RaftSystemUtils.ts +489 -0
  216. package/src/RaftTypes.ts +279 -0
  217. package/src/RaftUpdateEvents.ts +48 -0
  218. package/src/RaftUpdateManager.ts +781 -0
  219. package/src/RaftUtils.ts +514 -0
  220. package/src/RaftWifiTypes.ts +36 -0
  221. package/src/main.ts +39 -0
  222. package/testdata/TestDeviceTypeRecs.json +492 -0
  223. package/tsconfig.json +30 -0
  224. package/tsconfig.react-native.json +29 -0
@@ -0,0 +1,51 @@
1
+ /////////////////////////////////////////////////////////////////////////////////////////////////////////////////
2
+ //
3
+ // RaftMsgTrackInfo
4
+ // Part of RaftJS
5
+ //
6
+ // Rob Dobson & Chris Greening 2020-2024
7
+ // (C) 2020-2024 All rights reserved
8
+ //
9
+ /////////////////////////////////////////////////////////////////////////////////////////////////////////////////
10
+
11
+ export class RaftMsgTrackInfo {
12
+ static readonly MAX_MSG_NUM = 255;
13
+ static readonly MSG_RESPONSE_TIMEOUT_MS = 2000;
14
+ static readonly MSG_RETRY_COUNT = 5;
15
+ msgOutstanding = false;
16
+ msgFrame: Uint8Array = new Uint8Array();
17
+ msgSentMs = 0;
18
+ retryCount = 0;
19
+ withResponse = false;
20
+ bridgeID: number | undefined = undefined;
21
+ msgHandle = 0;
22
+ msgTimeoutMs: number | undefined = undefined;
23
+ resolve: unknown;
24
+ reject: unknown;
25
+
26
+ constructor() {
27
+ this.msgOutstanding = false;
28
+ }
29
+
30
+ set(
31
+ msgOutstanding: boolean,
32
+ msgFrame: Uint8Array,
33
+ withResponse: boolean,
34
+ bridgeID: number | undefined = undefined,
35
+ msgHandle: number,
36
+ msgTimeoutMs: number | undefined,
37
+ resolve: unknown,
38
+ reject: unknown,
39
+ ) {
40
+ this.msgOutstanding = msgOutstanding;
41
+ this.msgFrame = msgFrame;
42
+ this.retryCount = 0;
43
+ this.msgSentMs = Date.now();
44
+ this.withResponse = withResponse;
45
+ this.bridgeID = bridgeID;
46
+ this.msgHandle = msgHandle;
47
+ this.msgTimeoutMs = msgTimeoutMs;
48
+ this.resolve = resolve;
49
+ this.reject = reject;
50
+ }
51
+ }
@@ -0,0 +1,46 @@
1
+ /////////////////////////////////////////////////////////////////////////////////////////////////////////////////
2
+ //
3
+ // Raft Protocol Definitions
4
+ // Part of RaftJS
5
+ //
6
+ // Rob Dobson & Chris Greening 2020-2024
7
+ // (C) 2020-2024 All rights reserved
8
+ //
9
+ /////////////////////////////////////////////////////////////////////////////////////////////////////////////////
10
+
11
+ // RIC Protocols
12
+ export const RICSERIAL_MSG_NUM_POS = 0;
13
+ export const RICSERIAL_PROTOCOL_POS = 1;
14
+ export const RICSERIAL_PAYLOAD_POS = 2;
15
+ export const RICREST_REST_ELEM_CODE_POS = 0;
16
+ export const RICREST_HEADER_PAYLOAD_POS = 1;
17
+ export const RICREST_FILEBLOCK_CHANNEL_POS = 0;
18
+ export const RICREST_FILEBLOCK_FILEPOS_POS = 0;
19
+ export const RICREST_FILEBLOCK_FILEPOS_POS_BYTES = 4;
20
+ export const RICREST_FILEBLOCK_PAYLOAD_POS = 4;
21
+ export const RICREST_BRIDGE_ID_POS = 2;
22
+ export const RICREST_BRIDGE_PAYLOAD_POS = 3;
23
+
24
+ // Protocol enums
25
+ export enum RICRESTElemCode {
26
+ RICREST_ELEM_CODE_URL,
27
+ RICREST_ELEM_CODE_CMDRESPJSON,
28
+ RICREST_ELEM_CODE_BODY,
29
+ RICREST_ELEM_CODE_COMMAND_FRAME,
30
+ RICREST_ELEM_CODE_FILEBLOCK,
31
+ }
32
+
33
+ export enum RaftCommsMsgTypeCode {
34
+ MSG_TYPE_COMMAND,
35
+ MSG_TYPE_RESPONSE,
36
+ MSG_TYPE_PUBLISH,
37
+ MSG_TYPE_REPORT,
38
+ }
39
+
40
+ export enum RaftCommsMsgProtocol {
41
+ MSG_PROTOCOL_ROSSERIAL = 0,
42
+ MSG_PROTOCOL_RESERVED_1 = 1,
43
+ MSG_PROTOCOL_RICREST = 2,
44
+ MSG_PROTOCOL_BRIDGE_RICREST = 3,
45
+ }
46
+
@@ -0,0 +1,329 @@
1
+ /////////////////////////////////////////////////////////////////////////////////////////////////////////////////
2
+ //
3
+ // RafStreamHandler
4
+ // Part of RaftJS
5
+ //
6
+ // Rob Dobson & Chris Greening 2020-2024
7
+ // (C) 2020-2024 All rights reserved
8
+ //
9
+ /////////////////////////////////////////////////////////////////////////////////////////////////////////////////
10
+
11
+ import RaftLog from './RaftLog'
12
+ import RaftMsgHandler from './RaftMsgHandler';
13
+ import RaftCommsStats from './RaftCommsStats';
14
+ import { RaftOKFail, RaftStreamStartResp, RaftStreamType } from './RaftTypes';
15
+ import RaftConnector from './RaftConnector';
16
+ import { RaftConnEvent } from './RaftConnEvents';
17
+ import { RICRESTElemCode } from './RaftProtocolDefs'
18
+ import RaftUtils from './RaftUtils';
19
+
20
+ export default class RaftStreamHandler {
21
+
22
+ // Queue of audio stream requests
23
+ /*
24
+ private _streamAudioQueue: {
25
+ streamContents: Uint8Array;
26
+ audioDuration: number;
27
+ }[] = [];
28
+ */
29
+
30
+ // Stream state
31
+ private _streamID: number | null = null;
32
+ DEFAULT_MAX_BLOCK_SIZE = 475;
33
+ private _maxBlockSize: number = this.DEFAULT_MAX_BLOCK_SIZE;
34
+
35
+ // Handler of messages
36
+ private _msgHandler: RaftMsgHandler;
37
+
38
+ // RaftCommsStats
39
+ private _commsStats: RaftCommsStats;
40
+
41
+ // RaftConnector
42
+ private _raftConnector: RaftConnector;
43
+
44
+ // Flow control
45
+ private _soktoReceived = false;
46
+ private _soktoPos = 0;
47
+
48
+ private _streamIsStarting = false;
49
+ private _lastStreamStartTime = 0;
50
+
51
+ private _isStreaming = false;
52
+ private _isPaused = false;
53
+ private _streamBuffer: Uint8Array<ArrayBuffer> = new Uint8Array();
54
+ private _audioDuration = 0;
55
+ private _audioByteRate = 0;
56
+ private _streamPos = 0;
57
+ private _numBlocksWithoutPause = 15;
58
+ private _legacySoktoMode = false;
59
+
60
+
61
+ // soundFinishPoint timer
62
+ private soundFinishPoint: NodeJS.Timeout | null = null;
63
+
64
+ constructor(msgHandler: RaftMsgHandler, commsStats: RaftCommsStats, raftConnector: RaftConnector) {
65
+ this._raftConnector = raftConnector;
66
+ this._msgHandler = msgHandler;
67
+ this._commsStats = commsStats;
68
+ this.onSoktoMsg = this.onSoktoMsg.bind(this);
69
+ }
70
+
71
+ setNumBlocksWithoutPause(numBlocks: number) {
72
+ this._numBlocksWithoutPause = numBlocks;
73
+ }
74
+
75
+ setLegacySoktoMode(legacyMode: boolean) {
76
+ RaftLog.debug(`Setting legacy sokto mode to ${legacyMode}`);
77
+ this._legacySoktoMode = legacyMode;
78
+ }
79
+
80
+ // Start streaming audio
81
+ streamAudio(streamContents: Uint8Array, clearExisting: boolean, audioDuration: number): void {
82
+ if (!clearExisting)
83
+ RaftLog.debug(`only clearExisting = true is supported right now.`);
84
+
85
+ // TODO - if clearExisting is not set, form a queue
86
+ if (this._streamIsStarting || this._lastStreamStartTime > (Date.now() - 500)) {
87
+ RaftLog.warn(`Unable to start sound, too soon since last request`);
88
+ return;
89
+ }
90
+
91
+ this._isPaused = true;
92
+ this._streamIsStarting = true;
93
+ this._lastStreamStartTime = Date.now();
94
+
95
+ this._soktoReceived = false;
96
+ this._soktoPos = 0;
97
+ this._streamPos = 0;
98
+ this._streamBuffer = RaftUtils.toArrayBufferView(streamContents);
99
+ this._audioDuration = audioDuration;
100
+ this._audioByteRate = (streamContents.length / audioDuration) * 1000;
101
+
102
+ this.clearFinishPointTimeout();
103
+
104
+ this._sendStreamStartMsg("audio.mp3", "streamaudio", RaftStreamType.REAL_TIME_STREAM, streamContents).then(
105
+ (result: boolean) => {
106
+ this._isPaused = false;
107
+ this._streamIsStarting = false;
108
+ if (!result) {
109
+ RaftLog.warn(`Unable to start stream. ufStart message send failed`);
110
+ return;
111
+ }
112
+ //this.streamingPerformanceChecker();
113
+ if (!this._isStreaming) {
114
+ this._isStreaming = true;
115
+ this._sendStreamBuffer();
116
+ }
117
+
118
+ }
119
+ );
120
+ }
121
+
122
+ async streamCancel(): Promise<void> {
123
+ this._streamBuffer = new Uint8Array();
124
+ this.clearFinishPointTimeout();
125
+ }
126
+
127
+ public isStreamStarting() {
128
+ return this._streamIsStarting;
129
+ }
130
+
131
+
132
+ clearFinishPointTimeout() {
133
+ if (this.soundFinishPoint) {
134
+ clearTimeout(this.soundFinishPoint);
135
+ this.soundFinishPoint = null;
136
+ }
137
+ }
138
+
139
+ streamingPerformanceChecker() {
140
+ if (this._audioDuration) {
141
+ this.clearFinishPointTimeout();
142
+ this.soundFinishPoint = setTimeout(() => {
143
+ // if the streaming hasn't finished before the end of the audio
144
+ // we can assume we are having streaming issues
145
+
146
+ // publish event in case we are having issues
147
+ this._raftConnector.onConnEvent(RaftConnEvent.CONN_STREAMING_ISSUE);
148
+
149
+ this.clearFinishPointTimeout();
150
+ }, this._audioDuration + 500);
151
+ }
152
+ }
153
+
154
+ // Send the start message
155
+ private async _sendStreamStartMsg(
156
+ streamName: string,
157
+ targetEndpoint: string,
158
+ streamTypeEnum: RaftStreamType,
159
+ streamContents: Uint8Array,
160
+ ): Promise<boolean> {
161
+ // Stream start command message
162
+ const streamType = 'rtstream';
163
+ const cmdMsg = `{"cmdName":"ufStart","reqStr":"ufStart","fileType":"${streamType}","fileName":"${streamName}","endpoint":"${targetEndpoint}","fileLen":${streamContents.length}}`;
164
+
165
+ // Debug
166
+ RaftLog.debug(`sendStreamStartMsg ${cmdMsg}`);
167
+
168
+ // Send
169
+ let streamStartResp = null;
170
+ try {
171
+ streamStartResp = await this._msgHandler.sendRICREST<RaftStreamStartResp>(
172
+ cmdMsg,
173
+ RICRESTElemCode.RICREST_ELEM_CODE_COMMAND_FRAME,
174
+ );
175
+ } catch (err) {
176
+ RaftLog.warn(`sendStreamStartMsg error ${err}`);
177
+ return false;
178
+ }
179
+
180
+ // Extract params
181
+ if (streamStartResp && (streamStartResp.rslt === 'ok')) {
182
+ this._streamID = streamStartResp.streamID;
183
+ this._maxBlockSize = streamStartResp.maxBlockSize || this.DEFAULT_MAX_BLOCK_SIZE;
184
+ this.streamingPerformanceChecker();
185
+ RaftLog.verbose(
186
+ `sendStreamStartMsg streamID ${this._streamID} maxBlockSize ${this._maxBlockSize} streamType ${streamTypeEnum}`,
187
+ );
188
+ } else {
189
+ RaftLog.warn(`sendStreamStartMsg failed ${streamStartResp ? streamStartResp.rslt : 'no response'}`);
190
+ return false;
191
+ }
192
+ return true;
193
+ }
194
+
195
+ get maxBlockSize() {
196
+ return this._maxBlockSize;
197
+ }
198
+
199
+ set maxBlockSize(maxBlockSize: number) {
200
+ this._maxBlockSize = maxBlockSize;
201
+ this.DEFAULT_MAX_BLOCK_SIZE = maxBlockSize;
202
+ }
203
+
204
+ private async _sendStreamEndMsg(
205
+ streamID: number | null,
206
+ ): Promise<boolean> {
207
+ if (streamID === null) {
208
+ return false;
209
+ }
210
+ // Stram end command message
211
+ const cmdMsg = `{"cmdName":"ufEnd","reqStr":"ufEnd","streamID":${streamID}}`;
212
+
213
+ // Debug
214
+ RaftLog.debug(`sendStreamEndMsg ${cmdMsg}`);
215
+
216
+ // Send
217
+ let streamEndResp = null;
218
+ try {
219
+ streamEndResp = await this._msgHandler.sendRICREST<RaftOKFail>(
220
+ cmdMsg,
221
+ RICRESTElemCode.RICREST_ELEM_CODE_COMMAND_FRAME,
222
+ );
223
+ } catch (err) {
224
+ RaftLog.warn(`sendStreamEndMsg error ${err}`);
225
+ return false;
226
+ }
227
+ return streamEndResp.rslt === 'ok';
228
+ }
229
+
230
+ /*
231
+ private async _sendAudioStopMsg(): Promise<RaftOKFail> {
232
+ const cmdMsg = `{"cmdName":"audio/stop"}`;
233
+
234
+ // Debug
235
+ RaftLog.debug(`sendAudioStopMsg ${cmdMsg}`);
236
+
237
+ // Send
238
+ return this._msgHandler.sendRICREST<RaftOKFail>(
239
+ cmdMsg,
240
+ RICRESTElemCode.RICREST_ELEM_CODE_COMMAND_FRAME,
241
+ );
242
+ }
243
+
244
+
245
+ private async _sendStreamCancelMsg(): Promise<RaftOKFail> {
246
+ // File cancel command message
247
+ const cmdMsg = `{"cmdName":"ufCancel","reqStr":"ufCancel","streamID":${this._streamID}}`;
248
+
249
+ // Debug
250
+ RaftLog.debug(`sendStreamCancelMsg ${cmdMsg}`);
251
+
252
+ // Send
253
+ return this._msgHandler.sendRICREST<RaftOKFail>(
254
+ cmdMsg,
255
+ RICRESTElemCode.RICREST_ELEM_CODE_COMMAND_FRAME,
256
+ );
257
+ }
258
+ */
259
+
260
+ private async _sendStreamBuffer(): Promise<boolean> {
261
+ const streamStartTime = Date.now();
262
+
263
+ // Check streamID is valid
264
+ if (this._streamID === null) {
265
+ return false;
266
+ }
267
+
268
+ let blockNum = 0;
269
+ // Send stream blocks
270
+ while (this._soktoPos < this._streamBuffer.length || this._isPaused) {
271
+ if (this._isPaused) {
272
+ await new Promise((resolve) => setTimeout(resolve, 5));
273
+ continue;
274
+ }
275
+
276
+ // Check for new sokto
277
+ if (this._soktoReceived) {
278
+ if (this._legacySoktoMode)
279
+ this._streamPos = this._soktoPos;
280
+ // apart from when in legacy mode, the sokto message is now informational only,
281
+ // to allow the central to slow down sending of data if it is swamping the peripheral
282
+ RaftLog.verbose(`sendStreamContents ${Date.now() - streamStartTime}ms soktoReceived for ${this._streamPos}`);
283
+ this._soktoReceived = false;
284
+
285
+ // receiving an sokto message before the completion of the stream means that the streaming is not keeping up
286
+ this._raftConnector.onConnEvent(RaftConnEvent.CONN_STREAMING_ISSUE);
287
+ }
288
+
289
+ // Send stream block
290
+ const blockSize = Math.min(this._streamBuffer.length - this._streamPos, this._maxBlockSize);
291
+ const block = this._streamBuffer.slice(this._streamPos, this._streamPos + blockSize);
292
+ if (block.length > 0) {
293
+ const sentOk = await this._msgHandler.sendStreamBlock(block, this._streamPos, this._streamID);
294
+ this._commsStats.recordStreamBytes(block.length);
295
+
296
+ RaftLog.verbose(
297
+ `sendStreamContents ${sentOk ? "OK" : "FAILED"} ${Date.now() - streamStartTime}ms pos ${this._streamPos} ${blockSize} ${block.length} ${this._soktoPos}`,
298
+ );
299
+ if (!sentOk) {
300
+ return false;
301
+ }
302
+ this._streamPos += blockSize;
303
+ blockNum += 1;
304
+
305
+ if (this._audioByteRate && blockNum > this._numBlocksWithoutPause) {
306
+ const pauseTime = ((blockSize / this._audioByteRate) * 1000) - 10;
307
+ RaftLog.verbose(`Pausing for ${pauseTime} ms between audio packets. Bit rate ${this._audioByteRate * 8}`)
308
+ await new Promise((resolve) => setTimeout(resolve, pauseTime));
309
+ }
310
+ }
311
+
312
+ // Wait to ensure we don't hog the CPU
313
+ await new Promise((resolve) => setTimeout(resolve, 1));
314
+ }
315
+
316
+ this._isStreaming = false;
317
+ this.clearFinishPointTimeout();
318
+ await this._sendStreamEndMsg(this._streamID);
319
+
320
+ return true;
321
+ }
322
+
323
+ onSoktoMsg(soktoPos: number) {
324
+ // Get how far we've progressed in file
325
+ this._soktoPos = soktoPos;
326
+ this._soktoReceived = true;
327
+ RaftLog.debug(`onSoktoMsg received file up to ${this._soktoPos}`);
328
+ }
329
+ }