@webex/plugin-meetings 3.0.0-beta.27 → 3.0.0-beta.271

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 (362) hide show
  1. package/README.md +46 -8
  2. package/dist/annotation/annotation.types.js +7 -0
  3. package/dist/annotation/annotation.types.js.map +1 -0
  4. package/dist/annotation/constants.js +49 -0
  5. package/dist/annotation/constants.js.map +1 -0
  6. package/dist/annotation/index.js +342 -0
  7. package/dist/annotation/index.js.map +1 -0
  8. package/dist/breakouts/breakout.js +114 -14
  9. package/dist/breakouts/breakout.js.map +1 -1
  10. package/dist/breakouts/edit-lock-error.js +52 -0
  11. package/dist/breakouts/edit-lock-error.js.map +1 -0
  12. package/dist/breakouts/events.js +45 -0
  13. package/dist/breakouts/events.js.map +1 -0
  14. package/dist/breakouts/index.js +763 -31
  15. package/dist/breakouts/index.js.map +1 -1
  16. package/dist/breakouts/request.js +78 -0
  17. package/dist/breakouts/request.js.map +1 -0
  18. package/dist/breakouts/utils.js +67 -0
  19. package/dist/breakouts/utils.js.map +1 -0
  20. package/dist/common/errors/no-meeting-info.js +51 -0
  21. package/dist/common/errors/no-meeting-info.js.map +1 -0
  22. package/dist/common/errors/webex-errors.js +28 -7
  23. package/dist/common/errors/webex-errors.js.map +1 -1
  24. package/dist/common/logs/logger-proxy.js +1 -1
  25. package/dist/common/logs/logger-proxy.js.map +1 -1
  26. package/dist/common/queue.js +24 -9
  27. package/dist/common/queue.js.map +1 -1
  28. package/dist/config.js +5 -10
  29. package/dist/config.js.map +1 -1
  30. package/dist/constants.js +203 -28
  31. package/dist/constants.js.map +1 -1
  32. package/dist/controls-options-manager/enums.js +14 -2
  33. package/dist/controls-options-manager/enums.js.map +1 -1
  34. package/dist/controls-options-manager/index.js +109 -15
  35. package/dist/controls-options-manager/index.js.map +1 -1
  36. package/dist/controls-options-manager/types.js +7 -0
  37. package/dist/controls-options-manager/types.js.map +1 -0
  38. package/dist/controls-options-manager/util.js +309 -18
  39. package/dist/controls-options-manager/util.js.map +1 -1
  40. package/dist/index.js +112 -1
  41. package/dist/index.js.map +1 -1
  42. package/dist/interpretation/collection.js +23 -0
  43. package/dist/interpretation/collection.js.map +1 -0
  44. package/dist/interpretation/index.js +366 -0
  45. package/dist/interpretation/index.js.map +1 -0
  46. package/dist/interpretation/siLanguage.js +25 -0
  47. package/dist/interpretation/siLanguage.js.map +1 -0
  48. package/dist/locus-info/controlsUtils.js +91 -2
  49. package/dist/locus-info/controlsUtils.js.map +1 -1
  50. package/dist/locus-info/index.js +381 -62
  51. package/dist/locus-info/index.js.map +1 -1
  52. package/dist/locus-info/infoUtils.js +7 -1
  53. package/dist/locus-info/infoUtils.js.map +1 -1
  54. package/dist/locus-info/mediaSharesUtils.js +57 -1
  55. package/dist/locus-info/mediaSharesUtils.js.map +1 -1
  56. package/dist/locus-info/parser.js +224 -63
  57. package/dist/locus-info/parser.js.map +1 -1
  58. package/dist/locus-info/selfUtils.js +89 -14
  59. package/dist/locus-info/selfUtils.js.map +1 -1
  60. package/dist/media/index.js +58 -116
  61. package/dist/media/index.js.map +1 -1
  62. package/dist/media/properties.js +72 -123
  63. package/dist/media/properties.js.map +1 -1
  64. package/dist/meeting/in-meeting-actions.js +82 -2
  65. package/dist/meeting/in-meeting-actions.js.map +1 -1
  66. package/dist/meeting/index.js +3118 -2814
  67. package/dist/meeting/index.js.map +1 -1
  68. package/dist/meeting/locusMediaRequest.js +292 -0
  69. package/dist/meeting/locusMediaRequest.js.map +1 -0
  70. package/dist/meeting/muteState.js +230 -124
  71. package/dist/meeting/muteState.js.map +1 -1
  72. package/dist/meeting/request.js +256 -196
  73. package/dist/meeting/request.js.map +1 -1
  74. package/dist/meeting/util.js +601 -417
  75. package/dist/meeting/util.js.map +1 -1
  76. package/dist/meeting-info/index.js +70 -7
  77. package/dist/meeting-info/index.js.map +1 -1
  78. package/dist/meeting-info/meeting-info-v2.js +189 -51
  79. package/dist/meeting-info/meeting-info-v2.js.map +1 -1
  80. package/dist/meeting-info/util.js +1 -1
  81. package/dist/meeting-info/util.js.map +1 -1
  82. package/dist/meeting-info/utilv2.js +36 -36
  83. package/dist/meeting-info/utilv2.js.map +1 -1
  84. package/dist/meetings/collection.js +22 -0
  85. package/dist/meetings/collection.js.map +1 -1
  86. package/dist/meetings/index.js +394 -94
  87. package/dist/meetings/index.js.map +1 -1
  88. package/dist/meetings/meetings.types.js +7 -0
  89. package/dist/meetings/meetings.types.js.map +1 -0
  90. package/dist/meetings/request.js +2 -0
  91. package/dist/meetings/request.js.map +1 -1
  92. package/dist/meetings/util.js +71 -1
  93. package/dist/meetings/util.js.map +1 -1
  94. package/dist/member/index.js +49 -0
  95. package/dist/member/index.js.map +1 -1
  96. package/dist/member/types.js +25 -0
  97. package/dist/member/types.js.map +1 -0
  98. package/dist/member/util.js +121 -25
  99. package/dist/member/util.js.map +1 -1
  100. package/dist/members/collection.js +10 -0
  101. package/dist/members/collection.js.map +1 -1
  102. package/dist/members/index.js +86 -5
  103. package/dist/members/index.js.map +1 -1
  104. package/dist/members/request.js +106 -38
  105. package/dist/members/request.js.map +1 -1
  106. package/dist/members/types.js +15 -0
  107. package/dist/members/types.js.map +1 -0
  108. package/dist/members/util.js +316 -233
  109. package/dist/members/util.js.map +1 -1
  110. package/dist/metrics/constants.js +12 -5
  111. package/dist/metrics/constants.js.map +1 -1
  112. package/dist/metrics/index.js +1 -468
  113. package/dist/metrics/index.js.map +1 -1
  114. package/dist/multistream/mediaRequestManager.js +238 -49
  115. package/dist/multistream/mediaRequestManager.js.map +1 -1
  116. package/dist/multistream/receiveSlot.js +49 -16
  117. package/dist/multistream/receiveSlot.js.map +1 -1
  118. package/dist/multistream/receiveSlotManager.js +52 -34
  119. package/dist/multistream/receiveSlotManager.js.map +1 -1
  120. package/dist/multistream/remoteMedia.js +44 -18
  121. package/dist/multistream/remoteMedia.js.map +1 -1
  122. package/dist/multistream/remoteMediaGroup.js +60 -3
  123. package/dist/multistream/remoteMediaGroup.js.map +1 -1
  124. package/dist/multistream/remoteMediaManager.js +209 -59
  125. package/dist/multistream/remoteMediaManager.js.map +1 -1
  126. package/dist/multistream/sendSlotManager.js +233 -0
  127. package/dist/multistream/sendSlotManager.js.map +1 -0
  128. package/dist/reachability/index.js +225 -59
  129. package/dist/reachability/index.js.map +1 -1
  130. package/dist/reachability/request.js +17 -8
  131. package/dist/reachability/request.js.map +1 -1
  132. package/dist/reconnection-manager/index.js +199 -154
  133. package/dist/reconnection-manager/index.js.map +1 -1
  134. package/dist/recording-controller/index.js +21 -2
  135. package/dist/recording-controller/index.js.map +1 -1
  136. package/dist/recording-controller/util.js +9 -8
  137. package/dist/recording-controller/util.js.map +1 -1
  138. package/dist/roap/index.js +23 -29
  139. package/dist/roap/index.js.map +1 -1
  140. package/dist/roap/request.js +112 -97
  141. package/dist/roap/request.js.map +1 -1
  142. package/dist/roap/turnDiscovery.js +96 -36
  143. package/dist/roap/turnDiscovery.js.map +1 -1
  144. package/dist/rtcMetrics/constants.js +12 -0
  145. package/dist/rtcMetrics/constants.js.map +1 -0
  146. package/dist/rtcMetrics/index.js +117 -0
  147. package/dist/rtcMetrics/index.js.map +1 -0
  148. package/dist/statsAnalyzer/index.js +51 -34
  149. package/dist/statsAnalyzer/index.js.map +1 -1
  150. package/dist/statsAnalyzer/mqaUtil.js +6 -6
  151. package/dist/statsAnalyzer/mqaUtil.js.map +1 -1
  152. package/dist/types/annotation/annotation.types.d.ts +42 -0
  153. package/dist/types/annotation/constants.d.ts +31 -0
  154. package/dist/types/annotation/index.d.ts +117 -0
  155. package/dist/types/breakouts/edit-lock-error.d.ts +15 -0
  156. package/dist/types/breakouts/events.d.ts +8 -0
  157. package/dist/types/breakouts/request.d.ts +22 -0
  158. package/dist/types/breakouts/utils.d.ts +15 -0
  159. package/dist/types/common/errors/no-meeting-info.d.ts +14 -0
  160. package/dist/types/common/errors/webex-errors.d.ts +13 -1
  161. package/dist/types/common/queue.d.ts +9 -7
  162. package/dist/types/config.d.ts +1 -6
  163. package/dist/types/constants.d.ts +161 -21
  164. package/dist/types/controls-options-manager/enums.d.ts +11 -1
  165. package/dist/types/controls-options-manager/index.d.ts +17 -1
  166. package/dist/types/controls-options-manager/types.d.ts +43 -0
  167. package/dist/types/controls-options-manager/util.d.ts +1 -7
  168. package/dist/types/index.d.ts +6 -4
  169. package/dist/types/interpretation/collection.d.ts +5 -0
  170. package/dist/types/interpretation/index.d.ts +5 -0
  171. package/dist/types/interpretation/siLanguage.d.ts +5 -0
  172. package/dist/types/locus-info/index.d.ts +57 -4
  173. package/dist/types/locus-info/parser.d.ts +65 -6
  174. package/dist/types/media/index.d.ts +2 -0
  175. package/dist/types/media/properties.d.ts +34 -48
  176. package/dist/types/meeting/in-meeting-actions.d.ts +82 -2
  177. package/dist/types/meeting/index.d.ts +345 -507
  178. package/dist/types/meeting/locusMediaRequest.d.ts +74 -0
  179. package/dist/types/meeting/muteState.d.ts +99 -23
  180. package/dist/types/meeting/request.d.ts +72 -43
  181. package/dist/types/meeting/util.d.ts +101 -1
  182. package/dist/types/meeting-info/index.d.ts +13 -1
  183. package/dist/types/meeting-info/meeting-info-v2.d.ts +31 -1
  184. package/dist/types/meetings/collection.d.ts +8 -0
  185. package/dist/types/meetings/index.d.ts +88 -12
  186. package/dist/types/meetings/meetings.types.d.ts +4 -0
  187. package/dist/types/member/index.d.ts +13 -0
  188. package/dist/types/member/types.d.ts +32 -0
  189. package/dist/types/members/collection.d.ts +5 -0
  190. package/dist/types/members/index.d.ts +35 -2
  191. package/dist/types/members/request.d.ts +73 -9
  192. package/dist/types/members/types.d.ts +24 -0
  193. package/dist/types/members/util.d.ts +209 -1
  194. package/dist/types/metrics/constants.d.ts +11 -4
  195. package/dist/types/metrics/index.d.ts +4 -119
  196. package/dist/types/multistream/mediaRequestManager.d.ts +73 -5
  197. package/dist/types/multistream/receiveSlot.d.ts +16 -12
  198. package/dist/types/multistream/receiveSlotManager.d.ts +19 -4
  199. package/dist/types/multistream/remoteMedia.d.ts +8 -29
  200. package/dist/types/multistream/remoteMediaGroup.d.ts +0 -9
  201. package/dist/types/multistream/remoteMediaManager.d.ts +46 -2
  202. package/dist/types/multistream/sendSlotManager.d.ts +61 -0
  203. package/dist/types/reachability/index.d.ts +61 -7
  204. package/dist/types/reachability/request.d.ts +7 -3
  205. package/dist/types/reconnection-manager/index.d.ts +9 -0
  206. package/dist/types/recording-controller/index.d.ts +15 -1
  207. package/dist/types/recording-controller/util.d.ts +5 -4
  208. package/dist/types/roap/request.d.ts +15 -11
  209. package/dist/types/roap/turnDiscovery.d.ts +18 -1
  210. package/dist/types/rtcMetrics/constants.d.ts +4 -0
  211. package/dist/types/rtcMetrics/index.d.ts +47 -0
  212. package/dist/types/statsAnalyzer/index.d.ts +6 -1
  213. package/package.json +23 -20
  214. package/src/annotation/annotation.types.ts +50 -0
  215. package/src/annotation/constants.ts +36 -0
  216. package/src/annotation/index.ts +328 -0
  217. package/src/breakouts/README.md +44 -14
  218. package/src/breakouts/breakout.ts +87 -9
  219. package/src/breakouts/edit-lock-error.ts +25 -0
  220. package/src/breakouts/events.ts +56 -0
  221. package/src/breakouts/index.ts +646 -18
  222. package/src/breakouts/request.ts +55 -0
  223. package/src/breakouts/utils.ts +57 -0
  224. package/src/common/errors/no-meeting-info.ts +24 -0
  225. package/src/common/errors/webex-errors.ts +27 -2
  226. package/src/common/logs/logger-proxy.ts +1 -1
  227. package/src/common/queue.ts +22 -8
  228. package/src/config.ts +4 -9
  229. package/src/constants.ts +184 -18
  230. package/src/controls-options-manager/enums.ts +12 -0
  231. package/src/controls-options-manager/index.ts +116 -21
  232. package/src/controls-options-manager/types.ts +59 -0
  233. package/src/controls-options-manager/util.ts +294 -14
  234. package/src/index.ts +40 -0
  235. package/src/interpretation/README.md +60 -0
  236. package/src/interpretation/collection.ts +19 -0
  237. package/src/interpretation/index.ts +332 -0
  238. package/src/interpretation/siLanguage.ts +18 -0
  239. package/src/locus-info/controlsUtils.ts +108 -0
  240. package/src/locus-info/index.ts +412 -59
  241. package/src/locus-info/infoUtils.ts +10 -2
  242. package/src/locus-info/mediaSharesUtils.ts +64 -0
  243. package/src/locus-info/parser.ts +231 -39
  244. package/src/locus-info/selfUtils.ts +81 -5
  245. package/src/media/index.ts +100 -122
  246. package/src/media/properties.ts +85 -108
  247. package/src/meeting/in-meeting-actions.ts +163 -3
  248. package/src/meeting/index.ts +2526 -2309
  249. package/src/meeting/locusMediaRequest.ts +313 -0
  250. package/src/meeting/muteState.ts +229 -131
  251. package/src/meeting/request.ts +172 -121
  252. package/src/meeting/util.ts +588 -394
  253. package/src/meeting-info/index.ts +79 -8
  254. package/src/meeting-info/meeting-info-v2.ts +168 -14
  255. package/src/meeting-info/util.ts +1 -1
  256. package/src/meeting-info/utilv2.ts +23 -23
  257. package/src/meetings/collection.ts +20 -0
  258. package/src/meetings/index.ts +428 -108
  259. package/src/meetings/meetings.types.ts +12 -0
  260. package/src/meetings/request.ts +2 -0
  261. package/src/meetings/util.ts +79 -4
  262. package/src/member/index.ts +49 -0
  263. package/src/member/types.ts +38 -0
  264. package/src/member/util.ts +127 -25
  265. package/src/members/collection.ts +8 -0
  266. package/src/members/index.ts +107 -6
  267. package/src/members/request.ts +97 -17
  268. package/src/members/types.ts +28 -0
  269. package/src/members/util.ts +319 -240
  270. package/src/metrics/constants.ts +11 -4
  271. package/src/metrics/index.ts +1 -490
  272. package/src/multistream/mediaRequestManager.ts +289 -79
  273. package/src/multistream/receiveSlot.ts +55 -18
  274. package/src/multistream/receiveSlotManager.ts +46 -24
  275. package/src/multistream/remoteMedia.ts +27 -2
  276. package/src/multistream/remoteMediaGroup.ts +59 -0
  277. package/src/multistream/remoteMediaManager.ts +148 -30
  278. package/src/multistream/sendSlotManager.ts +170 -0
  279. package/src/reachability/index.ts +228 -37
  280. package/src/reachability/request.ts +17 -8
  281. package/src/reconnection-manager/index.ts +81 -54
  282. package/src/recording-controller/index.ts +20 -3
  283. package/src/recording-controller/util.ts +26 -9
  284. package/src/roap/index.ts +23 -30
  285. package/src/roap/request.ts +100 -104
  286. package/src/roap/turnDiscovery.ts +51 -25
  287. package/src/rtcMetrics/constants.ts +3 -0
  288. package/src/rtcMetrics/index.ts +100 -0
  289. package/src/statsAnalyzer/index.ts +73 -35
  290. package/src/statsAnalyzer/mqaUtil.ts +8 -10
  291. package/test/integration/spec/converged-space-meetings.js +60 -3
  292. package/test/integration/spec/journey.js +320 -261
  293. package/test/integration/spec/space-meeting.js +76 -3
  294. package/test/unit/spec/annotation/index.ts +418 -0
  295. package/test/unit/spec/breakouts/breakout.ts +142 -24
  296. package/test/unit/spec/breakouts/edit-lock-error.ts +30 -0
  297. package/test/unit/spec/breakouts/events.ts +89 -0
  298. package/test/unit/spec/breakouts/index.ts +1488 -67
  299. package/test/unit/spec/breakouts/request.ts +104 -0
  300. package/test/unit/spec/breakouts/utils.js +72 -0
  301. package/test/unit/spec/common/queue.js +31 -2
  302. package/test/unit/spec/controls-options-manager/index.js +163 -0
  303. package/test/unit/spec/controls-options-manager/util.js +576 -60
  304. package/test/unit/spec/fixture/locus.js +1 -0
  305. package/test/unit/spec/interpretation/collection.ts +15 -0
  306. package/test/unit/spec/interpretation/index.ts +589 -0
  307. package/test/unit/spec/interpretation/siLanguage.ts +28 -0
  308. package/test/unit/spec/locus-info/controlsUtils.js +316 -43
  309. package/test/unit/spec/locus-info/index.js +1283 -33
  310. package/test/unit/spec/locus-info/infoUtils.js +37 -15
  311. package/test/unit/spec/locus-info/mediaSharesUtils.ts +32 -0
  312. package/test/unit/spec/locus-info/parser.js +62 -22
  313. package/test/unit/spec/locus-info/selfConstant.js +27 -4
  314. package/test/unit/spec/locus-info/selfUtils.js +208 -17
  315. package/test/unit/spec/media/index.ts +104 -37
  316. package/test/unit/spec/meeting/in-meeting-actions.ts +81 -3
  317. package/test/unit/spec/meeting/index.js +4307 -1938
  318. package/test/unit/spec/meeting/locusMediaRequest.ts +442 -0
  319. package/test/unit/spec/meeting/muteState.js +408 -208
  320. package/test/unit/spec/meeting/request.js +440 -45
  321. package/test/unit/spec/meeting/utils.js +679 -64
  322. package/test/unit/spec/meeting-info/index.js +293 -0
  323. package/test/unit/spec/meeting-info/meetinginfov2.js +517 -5
  324. package/test/unit/spec/meeting-info/utilv2.js +21 -0
  325. package/test/unit/spec/meetings/collection.js +14 -0
  326. package/test/unit/spec/meetings/index.js +1007 -177
  327. package/test/unit/spec/meetings/utils.js +206 -2
  328. package/test/unit/spec/member/index.js +58 -4
  329. package/test/unit/spec/member/util.js +479 -35
  330. package/test/unit/spec/members/index.js +319 -1
  331. package/test/unit/spec/members/request.js +206 -27
  332. package/test/unit/spec/members/utils.js +184 -0
  333. package/test/unit/spec/metrics/index.js +1 -50
  334. package/test/unit/spec/multistream/mediaRequestManager.ts +803 -162
  335. package/test/unit/spec/multistream/receiveSlot.ts +72 -13
  336. package/test/unit/spec/multistream/receiveSlotManager.ts +58 -28
  337. package/test/unit/spec/multistream/remoteMedia.ts +30 -0
  338. package/test/unit/spec/multistream/remoteMediaGroup.ts +266 -0
  339. package/test/unit/spec/multistream/remoteMediaManager.ts +326 -0
  340. package/test/unit/spec/multistream/sendSlotManager.ts +242 -0
  341. package/test/unit/spec/reachability/index.ts +549 -9
  342. package/test/unit/spec/reachability/request.js +68 -0
  343. package/test/unit/spec/reconnection-manager/index.js +84 -9
  344. package/test/unit/spec/recording-controller/index.js +294 -218
  345. package/test/unit/spec/recording-controller/util.js +223 -96
  346. package/test/unit/spec/roap/index.ts +31 -51
  347. package/test/unit/spec/roap/request.ts +203 -85
  348. package/test/unit/spec/roap/turnDiscovery.ts +48 -13
  349. package/test/unit/spec/rtcMetrics/index.ts +68 -0
  350. package/test/unit/spec/stats-analyzer/index.js +29 -2
  351. package/test/utils/integrationTestUtils.js +46 -0
  352. package/test/utils/testUtils.js +0 -52
  353. package/dist/meeting/effectsState.js +0 -262
  354. package/dist/meeting/effectsState.js.map +0 -1
  355. package/dist/metrics/config.js +0 -299
  356. package/dist/metrics/config.js.map +0 -1
  357. package/dist/types/meeting/effectsState.d.ts +0 -42
  358. package/dist/types/metrics/config.d.ts +0 -178
  359. package/src/index.js +0 -16
  360. package/src/meeting/effectsState.ts +0 -211
  361. package/src/metrics/config.ts +0 -495
  362. package/test/unit/spec/meeting/effectsState.js +0 -285
@@ -13,7 +13,12 @@ var _defineProperty2 = _interopRequireDefault(require("@babel/runtime-corejs2/he
13
13
  var _difference2 = _interopRequireDefault(require("lodash/difference"));
14
14
  var _queue = _interopRequireDefault(require("../common/queue"));
15
15
  var _loggerProxy = _interopRequireDefault(require("../common/logs/logger-proxy"));
16
+ var _metrics = _interopRequireDefault(require("../metrics"));
17
+ var _constants = _interopRequireDefault(require("../metrics/constants"));
16
18
  var _templateObject, _templateObject2, _templateObject3, _templateObject4, _templateObject5, _templateObject6, _templateObject7, _templateObject8, _templateObject9, _templateObject10;
19
+ var MAX_OOO_DELTA_COUNT = 5; // when we receive an out-of-order delta and the queue builds up to MAX_OOO_DELTA_COUNT, we do a sync with Locus
20
+ var OOO_DELTA_WAIT_TIME = 10000; // [ms] minimum wait time before we do a sync if we get out-of-order deltas
21
+ var OOO_DELTA_WAIT_TIME_RANDOM_DELAY = 5000; // [ms] max random delay added to OOO_DELTA_WAIT_TIME
17
22
  /**
18
23
  * Locus Delta Parser
19
24
  * @private
@@ -22,6 +27,8 @@ var _templateObject, _templateObject2, _templateObject3, _templateObject4, _temp
22
27
  var Parser = /*#__PURE__*/function () {
23
28
  // processing status
24
29
 
30
+ // received an out-of-order delta, so waiting for the missing one
31
+
25
32
  // loci comparison states
26
33
 
27
34
  /**
@@ -29,24 +36,72 @@ var Parser = /*#__PURE__*/function () {
29
36
  */
30
37
  function Parser() {
31
38
  (0, _classCallCheck2.default)(this, Parser);
39
+ (0, _defineProperty2.default)(this, "status", void 0);
32
40
  (0, _defineProperty2.default)(this, "queue", void 0);
33
41
  (0, _defineProperty2.default)(this, "workingCopy", void 0);
34
- this.queue = new _queue.default();
35
- // @ts-ignore - This is declared as static class member and again being initialized here from same
36
- this.status = Parser.status.IDLE;
42
+ (0, _defineProperty2.default)(this, "syncTimer", void 0);
43
+ var deltaCompareFunc = function deltaCompareFunc(left, right) {
44
+ var _Parser$loci = Parser.loci,
45
+ LT = _Parser$loci.LT,
46
+ GT = _Parser$loci.GT;
47
+ var extract = Parser.extractComparisonState;
48
+ if (Parser.isSequenceEmpty(left)) {
49
+ return -1;
50
+ }
51
+ if (Parser.isSequenceEmpty(right)) {
52
+ return 1;
53
+ }
54
+ var result = extract(Parser.compareSequence(left.baseSequence, right.baseSequence));
55
+ if (result === LT) {
56
+ return -1;
57
+ }
58
+ if (result === GT) {
59
+ return 1;
60
+ }
61
+ return 0;
62
+ };
63
+ this.queue = new _queue.default(deltaCompareFunc);
64
+ this.status = 'IDLE';
37
65
  this.onDeltaAction = null;
38
66
  this.workingCopy = null;
67
+ this.syncTimer = null;
39
68
  }
40
69
 
41
70
  /**
42
- * Checks if two sequences overlap in time,
43
- * the sequence with the higher minimum value is greater.
44
- * Chooses sequence with most recent data.
45
- * @param {Types~Locus} current
46
- * @param {Types~Locus} incoming
47
- * @returns {string} loci comparison state
71
+ * Returns a debug string representing a locus delta - useful for logging
72
+ *
73
+ * @param {LocusDeltaDto} locus Locus delta
74
+ * @returns {string}
48
75
  */
49
76
  (0, _createClass2.default)(Parser, [{
77
+ key: "isNewFullLocus",
78
+ value:
79
+ /**
80
+ * Returns true if the incoming full locus DTO is newer than the current working copy
81
+ *
82
+ * @param {Types~Locus} incomingFullDto New Full Locus DTO
83
+ * @returns {string} either Parser.loci.USE_INCOMING or Parser.loci.USE_CURRENT
84
+ */
85
+ function isNewFullLocus(incomingFullDto) {
86
+ if (!Parser.isLoci(incomingFullDto)) {
87
+ _loggerProxy.default.logger.info('Locus-info:parser#isNewFullLocus --> Ignoring non-locus object.');
88
+ return false;
89
+ }
90
+ if (!this.workingCopy) {
91
+ // we don't have a working copy yet, so any full locus is better than nothing
92
+ return true;
93
+ }
94
+ var comparisonResult = Parser.compareFullDtoSequence(this.workingCopy, incomingFullDto);
95
+ return comparisonResult === Parser.loci.USE_INCOMING;
96
+ }
97
+
98
+ /**
99
+ * Compares Locus sequences
100
+ * @param {Types~Locus} current Current working copy
101
+ * @param {Types~Locus} incoming New Locus delta
102
+ * @returns {string}
103
+ */
104
+ }, {
50
105
  key: "isValidLocus",
51
106
  value:
52
107
  /**
@@ -55,19 +110,11 @@ var Parser = /*#__PURE__*/function () {
55
110
  * @returns {boolean}
56
111
  */
57
112
  function isValidLocus(newLoci) {
58
- var _this = this;
59
113
  var isValid = false;
60
- var IDLE = Parser.status.IDLE;
61
114
  var isLoci = Parser.isLoci;
62
- // @ts-ignore
63
- var setStatus = function setStatus(status) {
64
- // @ts-ignore
65
- _this.status = status;
66
- };
67
115
 
68
116
  // one or both objects are not locus delta events
69
117
  if (!isLoci(this.workingCopy) || !isLoci(newLoci)) {
70
- setStatus(IDLE);
71
118
  _loggerProxy.default.logger.info('Locus-info:parser#processDeltaEvent --> Ignoring non-locus object. workingCopy:', this.workingCopy, 'newLoci:', newLoci);
72
119
  } else {
73
120
  isValid = true;
@@ -89,18 +136,20 @@ var Parser = /*#__PURE__*/function () {
89
136
  * @returns {undefined}
90
137
  */
91
138
  function nextEvent() {
92
- // @ts-ignore
93
- if (this.status === Parser.status.PAUSED) {
139
+ if (this.status === 'PAUSED') {
94
140
  _loggerProxy.default.logger.info('Locus-info:parser#nextEvent --> Locus parser paused.');
95
141
  return;
96
142
  }
143
+ if (this.status === 'BLOCKED') {
144
+ _loggerProxy.default.logger.info('Locus-info:parser#nextEvent --> Locus parser blocked by out-of-order delta.');
145
+ return;
146
+ }
97
147
 
98
148
  // continue processing until queue is empty
99
149
  if (this.queue.size() > 0) {
100
150
  this.processDeltaEvent();
101
151
  } else {
102
- // @ts-ignore
103
- this.status = Parser.status.IDLE;
152
+ this.status = 'IDLE';
104
153
  }
105
154
  }
106
155
 
@@ -111,7 +160,7 @@ var Parser = /*#__PURE__*/function () {
111
160
  * @param {Types~Locus} locus Locus delta
112
161
  * @returns {undefined}
113
162
  */
114
- // eslint-disable-next-line no-unused-vars
163
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
115
164
  }, {
116
165
  key: "onDeltaAction",
117
166
  value: function onDeltaAction(action, locus) {}
@@ -126,14 +175,18 @@ var Parser = /*#__PURE__*/function () {
126
175
  value: function onDeltaEvent(loci) {
127
176
  // enqueue the new loci
128
177
  this.queue.enqueue(loci);
129
- // start processing events in the queue if idle
130
- // and a function handler is defined
131
- // @ts-ignore
132
- if (this.status === Parser.status.IDLE && this.onDeltaAction) {
133
- // Update status, ensure we only process one event at a time.
134
- // @ts-ignore
135
- this.status = Parser.status.WORKING;
136
- this.processDeltaEvent();
178
+ if (this.onDeltaAction) {
179
+ if (this.status === 'BLOCKED') {
180
+ if (this.queue.size() > MAX_OOO_DELTA_COUNT) {
181
+ this.triggerSync('queue too big, blocked on out-of-order delta');
182
+ } else {
183
+ this.processDeltaEvent();
184
+ }
185
+ } else if (this.status === 'IDLE') {
186
+ // Update status, ensure we only process one event at a time.
187
+ this.status = 'WORKING';
188
+ this.processDeltaEvent();
189
+ }
137
190
  }
138
191
  }
139
192
 
@@ -151,11 +204,61 @@ var Parser = /*#__PURE__*/function () {
151
204
  * @returns {undefined}
152
205
  */
153
206
  function pause() {
154
- // @ts-ignore
155
- this.status = Parser.status.PAUSED;
207
+ this.status = 'PAUSED';
156
208
  _loggerProxy.default.logger.info('Locus-info:parser#pause --> Locus parser paused.');
157
209
  }
158
210
 
211
+ /**
212
+ * Triggers a sync with Locus
213
+ *
214
+ * @param {string} reason used just for logging
215
+ * @returns {undefined}
216
+ */
217
+ }, {
218
+ key: "triggerSync",
219
+ value: function triggerSync(reason) {
220
+ _loggerProxy.default.logger.info("Locus-info:parser#triggerSync --> doing sync, reason: ".concat(reason));
221
+ this.stopSyncTimer();
222
+ this.pause();
223
+ this.onDeltaAction(Parser.loci.DESYNC, this.workingCopy);
224
+ }
225
+
226
+ /**
227
+ * Starts a timer with a random delay. When that timer expires we will do a sync.
228
+ *
229
+ * The main purpose of this timer is to handle a case when we get some out-of-order deltas,
230
+ * so we start waiting to receive the missing delta. If that delta never arrives, this timer
231
+ * will trigger a sync with Locus.
232
+ *
233
+ * @returns {undefined}
234
+ */
235
+ }, {
236
+ key: "startSyncTimer",
237
+ value: function startSyncTimer() {
238
+ var _this = this;
239
+ if (this.syncTimer === null) {
240
+ var timeout = OOO_DELTA_WAIT_TIME + Math.random() * OOO_DELTA_WAIT_TIME_RANDOM_DELAY;
241
+ this.syncTimer = setTimeout(function () {
242
+ _this.syncTimer = null;
243
+ _this.triggerSync('timer expired, blocked on out-of-order delta');
244
+ }, timeout);
245
+ }
246
+ }
247
+
248
+ /**
249
+ * Stops the timer for triggering a sync
250
+ *
251
+ * @returns {undefined}
252
+ */
253
+ }, {
254
+ key: "stopSyncTimer",
255
+ value: function stopSyncTimer() {
256
+ if (this.syncTimer !== null) {
257
+ clearTimeout(this.syncTimer);
258
+ this.syncTimer = null;
259
+ }
260
+ }
261
+
159
262
  /**
160
263
  * Processes next locus delta in the queue,
161
264
  * continues until the queue is empty
@@ -165,12 +268,14 @@ var Parser = /*#__PURE__*/function () {
165
268
  }, {
166
269
  key: "processDeltaEvent",
167
270
  value: function processDeltaEvent() {
168
- var _Parser$loci = Parser.loci,
169
- DESYNC = _Parser$loci.DESYNC,
170
- USE_INCOMING = _Parser$loci.USE_INCOMING;
271
+ var _Parser$loci2 = Parser.loci,
272
+ DESYNC = _Parser$loci2.DESYNC,
273
+ USE_INCOMING = _Parser$loci2.USE_INCOMING,
274
+ WAIT = _Parser$loci2.WAIT;
171
275
  var extract = Parser.extractComparisonState;
172
276
  var newLoci = this.queue.dequeue();
173
277
  if (!this.isValidLocus(newLoci)) {
278
+ this.nextEvent();
174
279
  return;
175
280
  }
176
281
  var result = Parser.compare(this.workingCopy, newLoci);
@@ -179,6 +284,7 @@ var Parser = /*#__PURE__*/function () {
179
284
  // limited debugging, use chrome extension
180
285
  // for full debugging.
181
286
  _loggerProxy.default.logger.debug("Locus-info:parser#processDeltaEvent --> Locus Debug: ".concat(result));
287
+ var needToWait = false;
182
288
  if (lociComparison === DESYNC) {
183
289
  // wait for desync response
184
290
  this.pause();
@@ -187,12 +293,26 @@ var Parser = /*#__PURE__*/function () {
187
293
  // Note: The working copy of parser gets updated in .onFullLocus()
188
294
  // and here when USE_INCOMING locus.
189
295
  this.workingCopy = newLoci;
296
+ } else if (lociComparison === WAIT) {
297
+ // we've taken newLoci from the front of the queue, so put it back there as we have to wait
298
+ // for the one that should be in front of it, before we can process it
299
+ this.queue.enqueue(newLoci);
300
+ needToWait = true;
301
+ }
302
+ if (needToWait) {
303
+ this.status = 'BLOCKED';
304
+ this.startSyncTimer();
305
+ } else {
306
+ this.stopSyncTimer();
307
+ if (this.status === 'BLOCKED') {
308
+ // we are not blocked anymore
309
+ this.status = 'WORKING';
310
+ _loggerProxy.default.logger.info("Locus-info:parser#processDeltaEvent --> received delta that we were waiting for ".concat(Parser.locus2string(newLoci), ", not blocked anymore"));
311
+ }
190
312
  }
191
313
  if (this.onDeltaAction) {
192
- _loggerProxy.default.logger.info("Locus-info:parser#processDeltaEvent --> Locus Delta Action: ".concat(lociComparison));
193
-
194
- // eslint-disable-next-line no-useless-call
195
- this.onDeltaAction.call(this, lociComparison, newLoci);
314
+ _loggerProxy.default.logger.info("Locus-info:parser#processDeltaEvent --> Locus Delta ".concat(Parser.locus2string(newLoci), ", Action: ").concat(lociComparison));
315
+ this.onDeltaAction(lociComparison, newLoci);
196
316
  }
197
317
  this.nextEvent();
198
318
  }
@@ -205,8 +325,7 @@ var Parser = /*#__PURE__*/function () {
205
325
  key: "resume",
206
326
  value: function resume() {
207
327
  _loggerProxy.default.logger.info('Locus-info:parser#resume --> Locus parser resumed.');
208
- // @ts-ignore
209
- this.status = Parser.status.WORKING;
328
+ this.status = 'WORKING';
210
329
  this.nextEvent();
211
330
  }
212
331
 
@@ -217,6 +336,24 @@ var Parser = /*#__PURE__*/function () {
217
336
  * @returns {object} Debug message
218
337
  */
219
338
  }], [{
339
+ key: "locus2string",
340
+ value: function locus2string(locus) {
341
+ var _locus$sequence;
342
+ if (!((_locus$sequence = locus.sequence) !== null && _locus$sequence !== void 0 && _locus$sequence.entries)) {
343
+ return 'invalid';
344
+ }
345
+ return locus.sequence.entries.length ? "seq=".concat(locus.sequence.entries.at(-1)) : 'empty';
346
+ }
347
+
348
+ /**
349
+ * Checks if two sequences overlap in time,
350
+ * the sequence with the higher minimum value is greater.
351
+ * Chooses sequence with most recent data.
352
+ * @param {Types~Locus} current
353
+ * @param {Types~Locus} incoming
354
+ * @returns {string} loci comparison state
355
+ */
356
+ }, {
220
357
  key: "checkSequenceOverlap",
221
358
  value: function checkSequenceOverlap(current, incoming) {
222
359
  var comparison = null;
@@ -381,12 +518,13 @@ var Parser = /*#__PURE__*/function () {
381
518
  }, {
382
519
  key: "compareDelta",
383
520
  value: function compareDelta(current, incoming) {
384
- var _Parser$loci2 = Parser.loci,
385
- LT = _Parser$loci2.LT,
386
- GT = _Parser$loci2.GT,
387
- EQ = _Parser$loci2.EQ,
388
- DESYNC = _Parser$loci2.DESYNC,
389
- USE_INCOMING = _Parser$loci2.USE_INCOMING;
521
+ var _Parser$loci3 = Parser.loci,
522
+ LT = _Parser$loci3.LT,
523
+ GT = _Parser$loci3.GT,
524
+ EQ = _Parser$loci3.EQ,
525
+ DESYNC = _Parser$loci3.DESYNC,
526
+ USE_INCOMING = _Parser$loci3.USE_INCOMING,
527
+ WAIT = _Parser$loci3.WAIT;
390
528
  var extract = Parser.extractComparisonState;
391
529
  var pack = Parser.packComparisonResult;
392
530
  var result = Parser.compareSequence(current.sequence, incoming.sequence);
@@ -400,6 +538,20 @@ var Parser = /*#__PURE__*/function () {
400
538
  case EQ:
401
539
  comparison = USE_INCOMING;
402
540
  break;
541
+ case LT:
542
+ if (extract(Parser.compareSequence(incoming.baseSequence, incoming.sequence)) === EQ) {
543
+ // special case where Locus sends a delta with baseSequence === sequence to trigger a sync,
544
+ // because the delta event is too large to be sent over mercury connection
545
+ comparison = DESYNC;
546
+ } else {
547
+ // the incoming locus has baseSequence from the future, so it is out-of-order,
548
+ // we are missing 1 or more locus that should be in front of it, we need to wait for it
549
+ comparison = WAIT;
550
+ _metrics.default.sendBehavioralMetric(_constants.default.LOCUS_DELTA_OUT_OF_ORDER, {
551
+ stack: new Error().stack
552
+ });
553
+ }
554
+ break;
403
555
  default:
404
556
  comparison = DESYNC;
405
557
  }
@@ -407,11 +559,24 @@ var Parser = /*#__PURE__*/function () {
407
559
  }
408
560
 
409
561
  /**
410
- * Compares Locus sequences
562
+ * Compares Locus sequences - it should be called only for full Locus DTOs, not deltas
563
+ *
411
564
  * @param {Types~Locus} current Current working copy
412
- * @param {Types~Locus} incoming New Locus delta
413
- * @returns {string}
565
+ * @param {Types~Locus} incomingFullDto New Full Locus DTO
566
+ * @returns {string} either Parser.loci.USE_INCOMING or Parser.loci.USE_CURRENT
414
567
  */
568
+ }, {
569
+ key: "compareFullDtoSequence",
570
+ value: function compareFullDtoSequence(current, incomingFullDto) {
571
+ if (Parser.isSequenceEmpty(current) || Parser.isSequenceEmpty(incomingFullDto)) {
572
+ return Parser.loci.USE_INCOMING;
573
+ }
574
+
575
+ // the sequence.entries list will always contain at least 1 entry
576
+ // https://sqbu-github.cisco.com/WebExSquared/cloud-apps/wiki/Locus-Sequence-Comparison-Algorithm
577
+
578
+ return incomingFullDto.sequence.entries.slice(-1)[0] > current.sequence.entries.slice(-1)[0] ? Parser.loci.USE_INCOMING : Parser.loci.USE_CURRENT;
579
+ }
415
580
  }, {
416
581
  key: "compareSequence",
417
582
  value: function compareSequence(current, incoming) {
@@ -452,14 +617,14 @@ var Parser = /*#__PURE__*/function () {
452
617
  }, {
453
618
  key: "compareToAction",
454
619
  value: function compareToAction(result) {
455
- var _Parser$loci3 = Parser.loci,
456
- DESYNC = _Parser$loci3.DESYNC,
457
- EQ = _Parser$loci3.EQ,
458
- ERROR = _Parser$loci3.ERROR,
459
- GT = _Parser$loci3.GT,
460
- LT = _Parser$loci3.LT,
461
- USE_CURRENT = _Parser$loci3.USE_CURRENT,
462
- USE_INCOMING = _Parser$loci3.USE_INCOMING;
620
+ var _Parser$loci4 = Parser.loci,
621
+ DESYNC = _Parser$loci4.DESYNC,
622
+ EQ = _Parser$loci4.EQ,
623
+ ERROR = _Parser$loci4.ERROR,
624
+ GT = _Parser$loci4.GT,
625
+ LT = _Parser$loci4.LT,
626
+ USE_CURRENT = _Parser$loci4.USE_CURRENT,
627
+ USE_INCOMING = _Parser$loci4.USE_INCOMING;
463
628
  var action = ERROR;
464
629
  switch (result) {
465
630
  case EQ:
@@ -675,11 +840,6 @@ var Parser = /*#__PURE__*/function () {
675
840
  return Parser;
676
841
  }();
677
842
  exports.default = Parser;
678
- (0, _defineProperty2.default)(Parser, "status", {
679
- IDLE: 'IDLE',
680
- PAUSED: 'PAUSED',
681
- WORKING: 'WORKING'
682
- });
683
843
  (0, _defineProperty2.default)(Parser, "loci", {
684
844
  EQ: 'EQUAL',
685
845
  GT: 'GREATER_THAN',
@@ -687,6 +847,7 @@ exports.default = Parser;
687
847
  DESYNC: 'DESYNC',
688
848
  USE_INCOMING: 'USE_INCOMING',
689
849
  USE_CURRENT: 'USE_CURRENT',
850
+ WAIT: 'WAIT',
690
851
  ERROR: 'ERROR'
691
852
  });
692
853
  //# sourceMappingURL=parser.js.map
@@ -1 +1 @@
1
- {"version":3,"names":["Parser","queue","SimpleQueue","status","IDLE","onDeltaAction","workingCopy","newLoci","isValid","isLoci","setStatus","LoggerProxy","logger","info","PAUSED","size","processDeltaEvent","action","locus","loci","enqueue","WORKING","DESYNC","USE_INCOMING","extract","extractComparisonState","dequeue","isValidLocus","result","compare","lociComparison","debug","pause","call","nextEvent","current","incoming","comparison","min","max","GT","LT","currentIsNotUnique","unique","length","incomingIsNotUnique","currentTotalRange","end","incomingTotalRange","EQ","currentIsUnique","incomingIsUnique","currentUniqueMin","incomingUniqueMin","currentHasNoRange","start","incomingHasNoRange","neitherSeqHasRange","hasUniqOverlap","list","some","seq","currentUniqOverlap","incomingUniqOverlap","debugInfo","isSequenceEmpty","pack","packComparisonResult","baseSequence","compareDelta","compareSequence","sequence","compareToAction","local","getMetaData","delta","getUniqueSequences","rules","checkSequenceOverlap","checkUnequalRanges","checkForUniqueEntries","checkIfOutOfSync","rule","ERROR","USE_CURRENT","lociComparisonResult","split","entries","first","last","slice","rangeStart","rangeEnd","baseLoci","otherLoci","diff","getNumbersOutOfRange","output","filter","num","sort","a","b","hasEmptyEntries","hasEmptyRange","hasProp","prop","Object","prototype","hasOwnProperty","newData","oldData","debugCode","mStr","strings","join","replace","resolutionMap","debugMap","SO001","title","description","logic","SO002","UR001","UR002","UR003","UE001","UE002","OOS001","OOS002","OOS003","debugObj","resolution"],"sources":["parser.ts"],"sourcesContent":["import {difference} from 'lodash';\n\nimport SimpleQueue from '../common/queue';\nimport LoggerProxy from '../common/logs/logger-proxy';\n\n/**\n * Locus Delta Parser\n * @private\n * https://sqbu-github.cisco.com/WebExSquared/cloud-apps/wiki/Locus-Delta-Events\n */\nexport default class Parser {\n // processing status\n static status = {\n IDLE: 'IDLE',\n PAUSED: 'PAUSED',\n WORKING: 'WORKING',\n };\n\n // loci comparison states\n static loci = {\n EQ: 'EQUAL',\n GT: 'GREATER_THAN',\n LT: 'LESS_THAN',\n DESYNC: 'DESYNC',\n USE_INCOMING: 'USE_INCOMING',\n USE_CURRENT: 'USE_CURRENT',\n ERROR: 'ERROR',\n };\n\n queue: any;\n workingCopy: any;\n\n /**\n * @constructs Parser\n */\n constructor() {\n this.queue = new SimpleQueue();\n // @ts-ignore - This is declared as static class member and again being initialized here from same\n this.status = Parser.status.IDLE;\n this.onDeltaAction = null;\n this.workingCopy = null;\n }\n\n /**\n * Checks if two sequences overlap in time,\n * the sequence with the higher minimum value is greater.\n * Chooses sequence with most recent data.\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {string} loci comparison state\n */\n static checkSequenceOverlap(current, incoming) {\n let comparison = null;\n\n // if earliest working copy sequence is more recent than last incoming sequence\n if (current.min > incoming.max) {\n // choose left side (current)\n comparison = `${Parser.loci.GT}:SO001`;\n }\n // if last working copy sequence is before the earliest incoming sequence\n else if (current.max < incoming.min) {\n // choose right side (incoming)\n comparison = `${Parser.loci.LT}:SO002`;\n }\n\n // if no match above, defaults to null\n return comparison;\n }\n\n /**\n * Checks if two sequences have unequal ranges.\n * Chooses sequence with most larger range.\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {object} loci comparison\n */\n static checkUnequalRanges(current, incoming) {\n let comparison = null;\n const currentIsNotUnique = current.unique.length === 0;\n const incomingIsNotUnique = incoming.unique.length === 0;\n const currentTotalRange = current.end - current.min;\n const incomingTotalRange = incoming.end - incoming.min;\n\n // no unique values for both loci\n if (currentIsNotUnique && incomingIsNotUnique) {\n // current working copy loci has a larger range\n if (currentTotalRange > incomingTotalRange) {\n // choose left side (current)\n comparison = `${Parser.loci.GT}:UR001`;\n }\n // incoming delta loci has a larger range\n else if (currentTotalRange < incomingTotalRange) {\n // choose right side (incoming)\n comparison = `${Parser.loci.LT}:UR002`;\n } else {\n // with no unique entries and with ranges either absent or\n // of the same size, the sequences are considered equal.\n comparison = `${Parser.loci.EQ}:UR003`;\n }\n }\n\n return comparison;\n }\n\n /**\n * Checks if either sequences has unique entries.\n * Entries are considered unique if they do not overlap\n * with other Loci sequences or range values.\n * Chooses sequence with the unique entries.\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {string} loci comparison state\n */\n static checkForUniqueEntries(current, incoming) {\n let comparison = null;\n const currentIsUnique = current.unique.length > 0;\n const incomingIsUnique = incoming.unique.length > 0;\n\n // current has unique entries and incoming does not\n if (currentIsUnique && !incomingIsUnique) {\n // choose left side (current)\n comparison = `${Parser.loci.GT}:UE001`;\n }\n // current has no unique entries but incoming does\n else if (!currentIsUnique && incomingIsUnique) {\n // choose right side (incoming)\n comparison = `${Parser.loci.LT}:UE002`;\n }\n\n return comparison;\n }\n\n /**\n * Checks both Locus Delta objects to see if they are\n * out of sync with one another. If so sends a DESYNC\n * request to the server.\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {string} loci comparison state\n */\n static checkIfOutOfSync(current, incoming) {\n let comparison = null;\n const currentUniqueMin = current.unique[0];\n const incomingUniqueMin = incoming.unique[0];\n\n const currentHasNoRange = !current.start && !current.end;\n const incomingHasNoRange = !incoming.start && !incoming.end;\n const neitherSeqHasRange = currentHasNoRange && incomingHasNoRange;\n\n const hasUniqOverlap = (list, min, max) => list.some((seq) => min < seq && seq < max);\n // current unique entries overlap the total range of incoming\n const currentUniqOverlap = hasUniqOverlap(current.unique, incoming.min, incoming.max);\n // vice-versa, incoming unique entries overlap the total range of current\n const incomingUniqOverlap = hasUniqOverlap(incoming.unique, current.min, current.max);\n\n if (neitherSeqHasRange || currentUniqOverlap || incomingUniqOverlap) {\n // outputs string indicating which condition occurred. ex: 0,1,0\n const debugInfo = `${+neitherSeqHasRange},${+currentUniqOverlap},${+incomingUniqOverlap}`;\n\n // send DESYNC to server\n comparison = `${Parser.loci.DESYNC}:OOS001:${debugInfo}`;\n } else if (currentUniqueMin > incomingUniqueMin) {\n // choose left side (current)\n comparison = `${Parser.loci.GT}:OOS002`;\n } else {\n // choose right side (incoming)\n comparison = `${Parser.loci.LT}:OOS003`;\n }\n\n return comparison;\n }\n\n /**\n * Compares two loci to determine which one contains the most recent state\n * @instance\n * @memberof Locus\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {string} loci comparison state\n */\n static compare(current, incoming) {\n const {isSequenceEmpty} = Parser;\n const {extractComparisonState: extract} = Parser;\n const {packComparisonResult: pack} = Parser;\n\n if (isSequenceEmpty(current) || isSequenceEmpty(incoming)) {\n return pack(Parser.loci.USE_INCOMING, 'C001');\n }\n\n if (incoming.baseSequence) {\n return pack(Parser.compareDelta(current, incoming), 'C002');\n }\n\n const result = Parser.compareSequence(current.sequence, incoming.sequence);\n const action = Parser.compareToAction(extract(result));\n\n return pack(action, result);\n }\n\n /**\n * Compares two loci sequences (with delta params) and indicates what action\n * to take.\n * @instance\n * @memberof Locus\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @private\n * @returns {string} loci comparison state\n */\n private static compareDelta(current, incoming) {\n const {LT, GT, EQ, DESYNC, USE_INCOMING} = Parser.loci;\n\n const {extractComparisonState: extract} = Parser;\n const {packComparisonResult: pack} = Parser;\n\n const result = Parser.compareSequence(current.sequence, incoming.sequence);\n let comparison = extract(result);\n\n if (comparison !== LT) {\n return pack(Parser.compareToAction(comparison), result);\n }\n\n comparison = Parser.compareSequence(current.sequence, incoming.baseSequence);\n\n switch (extract(comparison)) {\n case GT:\n case EQ:\n comparison = USE_INCOMING;\n break;\n\n default:\n comparison = DESYNC;\n }\n\n return pack(comparison, result);\n }\n\n /**\n * Compares Locus sequences\n * @param {Types~Locus} current Current working copy\n * @param {Types~Locus} incoming New Locus delta\n * @returns {string}\n */\n static compareSequence(current, incoming) {\n // Locus sequence comparison rules in order of priority.\n // https://sqbu-github.cisco.com/WebExSquared/cloud-apps/wiki/Locus-Sequence-Comparison-Algorithm\n\n const local: any = Parser.getMetaData(current);\n const delta: any = Parser.getMetaData(incoming);\n\n // update loci metadata\n local.unique = Parser.getUniqueSequences(local, delta);\n delta.unique = Parser.getUniqueSequences(delta, local);\n\n // Locus sequence comparison rules\n // order matters\n const rules = [\n Parser.checkSequenceOverlap,\n Parser.checkUnequalRanges,\n Parser.checkForUniqueEntries,\n Parser.checkIfOutOfSync,\n ];\n\n for (const rule of rules) {\n // Rule only returns a value if the rule applies,\n // otherwise returns null.\n const result = rule(local, delta);\n\n if (result) {\n return result;\n }\n }\n\n // error, none of rules above applied\n // should never get here as last rule\n // should be catch all.\n return Parser.loci.ERROR;\n }\n\n /**\n * Transates the result of a sequence comparison into an intended behavior\n * @param {string} result\n * @returns {string} Locus comparison action\n */\n static compareToAction(result: string) {\n const {DESYNC, EQ, ERROR, GT, LT, USE_CURRENT, USE_INCOMING} = Parser.loci;\n\n let action = ERROR;\n\n switch (result) {\n case EQ:\n case GT:\n action = USE_CURRENT;\n break;\n case LT:\n action = USE_INCOMING;\n break;\n case DESYNC:\n action = DESYNC;\n break;\n default:\n LoggerProxy.logger.info(\n `Locus-info:parser#compareToAction --> Error: ${result} is not a recognized sequence comparison result.`\n );\n }\n\n return action;\n }\n\n /**\n * Extracts a loci comparison from a string of data.\n * @param {string} lociComparisonResult Comparison result with extra data\n * @returns {string} Comparison of EQ, LT, GT, or DESYNC.\n */\n static extractComparisonState(lociComparisonResult: string) {\n return lociComparisonResult.split(':')[0];\n }\n\n /**\n * @typedef {object} LociMetadata\n * @property {number} start - Starting sequence number\n * @property {number} end - Ending sequence number\n * @property {number} first - First sequence number\n * @property {number} last - Last sequence number\n * @property {number} min - Minimum sequence number\n * @property {number} max - Maximum sequence number\n * @property {number} entries - Loci sequence entries\n */\n\n /**\n * Metadata for Locus delta\n * @param {Array.<number>} sequence Locus delta sequence\n * @returns {LociMetadata} Locus Delta Metadata\n */\n static getMetaData(sequence: any) {\n const {entries} = sequence;\n const first = entries[0];\n const last = entries.slice(-1)[0];\n\n // rangeStart or rangeEnd is 0 if a range doesn't exist\n const start = sequence.rangeStart;\n const end = sequence.rangeEnd;\n\n // sequence data\n return {\n start,\n end,\n first,\n last,\n // Rule is: rangeStart <= rangeEnd <= min(entries)\n min: start || first,\n // Grab last entry if exist else default to rangeEnd\n max: last || end,\n // keep reference to actual sequence entries\n entries,\n };\n }\n\n /**\n * Compares two Locus delta objects and notes unique\n * values contained within baseLoci.\n * @param {LociMetadata} baseLoci\n * @param {LociMetadata} otherLoci\n * @returns {Array.<number>} List of unique sequences\n */\n static getUniqueSequences(baseLoci: any, otherLoci: any) {\n const diff: any = difference(baseLoci.entries, otherLoci.entries);\n\n const {start, end} = otherLoci;\n\n return Parser.getNumbersOutOfRange(diff, start, end);\n }\n\n /**\n * Returns an array of numbers outside of a given range.\n * @param {Array.<number>} list Array to filter\n * @param {number} rangeStart Start of range\n * @param {number} rangeEnd End of range\n * @returns {Array.<number>} Array of numbers sorted ASC\n */\n static getNumbersOutOfRange(list: Array<number>, rangeStart: number, rangeEnd: number) {\n // Collect all numbers if number is outside of specified range\n const output = list.filter((num) => num < rangeStart || num > rangeEnd);\n\n // sort ascending\n return output.sort((a, b) => a - b);\n }\n\n /**\n * Checks if newLoci or workingCopy is invalid.\n * @param {Types~Locus} newLoci\n * @returns {boolean}\n */\n isValidLocus(newLoci) {\n let isValid = false;\n const {IDLE} = Parser.status;\n const {isLoci} = Parser;\n // @ts-ignore\n const setStatus = (status) => {\n // @ts-ignore\n this.status = status;\n };\n\n // one or both objects are not locus delta events\n if (!isLoci(this.workingCopy) || !isLoci(newLoci)) {\n setStatus(IDLE);\n LoggerProxy.logger.info(\n 'Locus-info:parser#processDeltaEvent --> Ignoring non-locus object. workingCopy:',\n this.workingCopy,\n 'newLoci:',\n newLoci\n );\n } else {\n isValid = true;\n }\n\n return isValid;\n }\n\n /**\n * Determines if a paricular locus's sequence is empty\n * @param {Types~Locus} locus\n * @returns {bool}\n */\n static isSequenceEmpty(locus) {\n const {sequence} = locus;\n const hasEmptyEntries = !sequence.entries?.length;\n const hasEmptyRange = sequence.rangeStart === 0 && sequence.rangeEnd === 0;\n\n return hasEmptyEntries && hasEmptyRange;\n }\n\n /**\n * Determines if an object has basic\n * structure of a locus object.\n * @param {Types~Locus} loci\n * @returns {boolean}\n */\n static isLoci(loci) {\n if (!loci || !loci.sequence) {\n return false;\n }\n const hasProp = (prop) => Object.prototype.hasOwnProperty.call(loci.sequence, prop);\n\n if (hasProp('rangeStart') && hasProp('rangeEnd')) {\n return true;\n }\n\n return false;\n }\n\n /**\n * Processes next event in queue,\n * if queue is empty sets status to idle.\n * @returns {undefined}\n */\n nextEvent() {\n // @ts-ignore\n if (this.status === Parser.status.PAUSED) {\n LoggerProxy.logger.info('Locus-info:parser#nextEvent --> Locus parser paused.');\n\n return;\n }\n\n // continue processing until queue is empty\n if (this.queue.size() > 0) {\n this.processDeltaEvent();\n } else {\n // @ts-ignore\n this.status = Parser.status.IDLE;\n }\n }\n\n /**\n * Function handler for delta actions,\n * is set by instance callee.\n * @param {string} action Locus delta action\n * @param {Types~Locus} locus Locus delta\n * @returns {undefined}\n */\n // eslint-disable-next-line no-unused-vars\n onDeltaAction(action: string, locus) {}\n\n /**\n * Event handler for locus delta events\n * @param {Types~Locus} loci Locus Delta\n * @returns {undefined}\n */\n onDeltaEvent(loci) {\n // enqueue the new loci\n this.queue.enqueue(loci);\n // start processing events in the queue if idle\n // and a function handler is defined\n // @ts-ignore\n if (this.status === Parser.status.IDLE && this.onDeltaAction) {\n // Update status, ensure we only process one event at a time.\n // @ts-ignore\n this.status = Parser.status.WORKING;\n\n this.processDeltaEvent();\n }\n }\n\n /**\n * Appends new data onto a string of existing data.\n * @param {string} newData\n * @param {string} oldData\n * @returns {string}\n */\n static packComparisonResult(newData: string, oldData: string) {\n return `${newData}:${oldData}`;\n }\n\n /**\n * Pause locus processing\n * @returns {undefined}\n */\n pause() {\n // @ts-ignore\n this.status = Parser.status.PAUSED;\n LoggerProxy.logger.info('Locus-info:parser#pause --> Locus parser paused.');\n }\n\n /**\n * Processes next locus delta in the queue,\n * continues until the queue is empty\n * or cleared.\n * @returns {undefined}\n */\n processDeltaEvent() {\n const {DESYNC, USE_INCOMING} = Parser.loci;\n const {extractComparisonState: extract} = Parser;\n const newLoci = this.queue.dequeue();\n\n if (!this.isValidLocus(newLoci)) {\n return;\n }\n\n const result = Parser.compare(this.workingCopy, newLoci);\n const lociComparison = extract(result);\n\n // limited debugging, use chrome extension\n // for full debugging.\n LoggerProxy.logger.debug(`Locus-info:parser#processDeltaEvent --> Locus Debug: ${result}`);\n\n if (lociComparison === DESYNC) {\n // wait for desync response\n this.pause();\n } else if (lociComparison === USE_INCOMING) {\n // update working copy for future comparisons.\n // Note: The working copy of parser gets updated in .onFullLocus()\n // and here when USE_INCOMING locus.\n this.workingCopy = newLoci;\n }\n\n if (this.onDeltaAction) {\n LoggerProxy.logger.info(\n `Locus-info:parser#processDeltaEvent --> Locus Delta Action: ${lociComparison}`\n );\n\n // eslint-disable-next-line no-useless-call\n this.onDeltaAction.call(this, lociComparison, newLoci);\n }\n\n this.nextEvent();\n }\n\n /**\n * Resume from a paused state\n * @returns {undefined}\n */\n resume() {\n LoggerProxy.logger.info('Locus-info:parser#resume --> Locus parser resumed.');\n // @ts-ignore\n this.status = Parser.status.WORKING;\n this.nextEvent();\n }\n\n /**\n * Gets related debug info for given error code\n * @param {string} debugCode Debug code\n * @param {string} comparison Locus comparison string\n * @returns {object} Debug message\n */\n static getDebugMessage(debugCode: string, comparison: string) {\n // removes extra spaces from multiline string\n const mStr = (strings) => strings.join('').replace(/\\s{2,}/g, ' ');\n\n const resolutionMap = {\n EQ: `${Parser.loci.LT}: is equal (current == incoming).`,\n LT: `${Parser.loci.LT}: choose right side (incoming).`,\n GT: `${Parser.loci.GT}: choose left side (current).`,\n };\n\n const debugMap = {\n SO001: {\n title: 'checkSequenceOverlap-001',\n description: mStr`Occurs if earliest working copy sequence is more \\\n recent than last incoming sequence.`,\n logic: 'current.min > incoming.max',\n },\n\n SO002: {\n title: 'checkSequenceOverlap-002',\n description: mStr`Occurs if last working copy sequence is before the \\\n earliest incoming sequence.`,\n logic: 'current.max < incoming.min',\n },\n\n UR001: {\n title: 'checkUnequalRanges-001',\n description: mStr`Occurs if there are no unique values for both loci, \\\n and the current working copy loci has a larger range.`,\n logic: 'currentTotalRange > incomingTotalRange',\n },\n\n UR002: {\n title: 'checkUnequalRanges-002',\n description: mStr`Occurs if there are no unique values for both loci, \\\n and the incoming delta loci has a larger range.`,\n logic: 'currentTotalRange < incomingTotalRange',\n },\n\n UR003: {\n title: 'checkUnequalRanges-003',\n description: mStr`Occurs if there are no unique values for both loci, \\\n and with ranges either absent or of the same size, the sequences \\\n are considered equal.`,\n logic: 'currentTotalRange == incomingTotalRange',\n },\n\n UE001: {\n title: 'checkForUniqueEntries-001',\n description: mStr`Occurs if current loci has unique entries and \\\n incoming does not. Entries are considered unique if they \\\n do not overlap with other Loci sequences or range values.`,\n logic: 'currentIsUnique && !incomingIsUnique',\n },\n\n UE002: {\n title: 'checkForUniqueEntries-002',\n description: mStr`Occurs if current has no unique entries but \\\n incoming does. Entries are considered unique if they \\\n do not overlap with other Loci sequences or range values.`,\n logic: '!currentIsUnique && incomingIsUnique',\n },\n\n OOS001: {\n title: 'checkIfOutOfSync-001',\n description: mStr`Occurs if neither sequence has a range, or \\\n if the current loci unique entries overlap the total range of the \\\n incoming sequence, or if the incoming unique entries overlap \\\n the total range of current sequence.`,\n logic: 'neitherSeqHasRange || currentUniqOverlap || incomingUniqOverlap',\n },\n\n OOS002: {\n title: 'checkIfOutOfSync-002',\n description: mStr`Occurs if the minimum value from sequences that are \\\n unique to the current loci is greater than the minimum value from \\\n sequences that are unique to the incoming loci.`,\n logic: 'currentUniqueMin > incomingUniqueMin',\n },\n\n OOS003: {\n title: 'checkIfOutOfSync-003',\n description: mStr`Occurs if none of the comparison rules applied. \\\n It is a catch all.`,\n logic: 'else (catch all)',\n },\n };\n\n const debugObj = debugMap[debugCode];\n\n debugObj.title = `Debug: ${debugObj.title}`;\n debugObj.resolution = resolutionMap[comparison];\n\n return debugObj;\n }\n}\n"],"mappings":";;;;;;;;;;;;;AAEA;AACA;AAAsD;AAEtD;AACA;AACA;AACA;AACA;AAJA,IAKqBA,MAAM;EACzB;;EAOA;;EAcA;AACF;AACA;EACE,kBAAc;IAAA;IAAA;IAAA;IACZ,IAAI,CAACC,KAAK,GAAG,IAAIC,cAAW,EAAE;IAC9B;IACA,IAAI,CAACC,MAAM,GAAGH,MAAM,CAACG,MAAM,CAACC,IAAI;IAChC,IAAI,CAACC,aAAa,GAAG,IAAI;IACzB,IAAI,CAACC,WAAW,GAAG,IAAI;EACzB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EAPE;IAAA;IAAA;IAyVA;AACF;AACA;AACA;AACA;IACE,sBAAaC,OAAO,EAAE;MAAA;MACpB,IAAIC,OAAO,GAAG,KAAK;MACnB,IAAOJ,IAAI,GAAIJ,MAAM,CAACG,MAAM,CAArBC,IAAI;MACX,IAAOK,MAAM,GAAIT,MAAM,CAAhBS,MAAM;MACb;MACA,IAAMC,SAAS,GAAG,SAAZA,SAAS,CAAIP,MAAM,EAAK;QAC5B;QACA,KAAI,CAACA,MAAM,GAAGA,MAAM;MACtB,CAAC;;MAED;MACA,IAAI,CAACM,MAAM,CAAC,IAAI,CAACH,WAAW,CAAC,IAAI,CAACG,MAAM,CAACF,OAAO,CAAC,EAAE;QACjDG,SAAS,CAACN,IAAI,CAAC;QACfO,oBAAW,CAACC,MAAM,CAACC,IAAI,CACrB,iFAAiF,EACjF,IAAI,CAACP,WAAW,EAChB,UAAU,EACVC,OAAO,CACR;MACH,CAAC,MAAM;QACLC,OAAO,GAAG,IAAI;MAChB;MAEA,OAAOA,OAAO;IAChB;;IAEA;AACF;AACA;AACA;AACA;EAJE;IAAA;IAAA;IAgCA;AACF;AACA;AACA;AACA;IACE,qBAAY;MACV;MACA,IAAI,IAAI,CAACL,MAAM,KAAKH,MAAM,CAACG,MAAM,CAACW,MAAM,EAAE;QACxCH,oBAAW,CAACC,MAAM,CAACC,IAAI,CAAC,sDAAsD,CAAC;QAE/E;MACF;;MAEA;MACA,IAAI,IAAI,CAACZ,KAAK,CAACc,IAAI,EAAE,GAAG,CAAC,EAAE;QACzB,IAAI,CAACC,iBAAiB,EAAE;MAC1B,CAAC,MAAM;QACL;QACA,IAAI,CAACb,MAAM,GAAGH,MAAM,CAACG,MAAM,CAACC,IAAI;MAClC;IACF;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;IACE;EAAA;IAAA;IAAA,OACA,uBAAca,MAAc,EAAEC,KAAK,EAAE,CAAC;;IAEtC;AACF;AACA;AACA;AACA;EAJE;IAAA;IAAA,OAKA,sBAAaC,IAAI,EAAE;MACjB;MACA,IAAI,CAAClB,KAAK,CAACmB,OAAO,CAACD,IAAI,CAAC;MACxB;MACA;MACA;MACA,IAAI,IAAI,CAAChB,MAAM,KAAKH,MAAM,CAACG,MAAM,CAACC,IAAI,IAAI,IAAI,CAACC,aAAa,EAAE;QAC5D;QACA;QACA,IAAI,CAACF,MAAM,GAAGH,MAAM,CAACG,MAAM,CAACkB,OAAO;QAEnC,IAAI,CAACL,iBAAiB,EAAE;MAC1B;IACF;;IAEA;AACF;AACA;AACA;AACA;AACA;EALE;IAAA;IAAA;IAUA;AACF;AACA;AACA;IACE,iBAAQ;MACN;MACA,IAAI,CAACb,MAAM,GAAGH,MAAM,CAACG,MAAM,CAACW,MAAM;MAClCH,oBAAW,CAACC,MAAM,CAACC,IAAI,CAAC,kDAAkD,CAAC;IAC7E;;IAEA;AACF;AACA;AACA;AACA;AACA;EALE;IAAA;IAAA,OAMA,6BAAoB;MAClB,mBAA+Bb,MAAM,CAACmB,IAAI;QAAnCG,MAAM,gBAANA,MAAM;QAAEC,YAAY,gBAAZA,YAAY;MAC3B,IAA+BC,OAAO,GAAIxB,MAAM,CAAzCyB,sBAAsB;MAC7B,IAAMlB,OAAO,GAAG,IAAI,CAACN,KAAK,CAACyB,OAAO,EAAE;MAEpC,IAAI,CAAC,IAAI,CAACC,YAAY,CAACpB,OAAO,CAAC,EAAE;QAC/B;MACF;MAEA,IAAMqB,MAAM,GAAG5B,MAAM,CAAC6B,OAAO,CAAC,IAAI,CAACvB,WAAW,EAAEC,OAAO,CAAC;MACxD,IAAMuB,cAAc,GAAGN,OAAO,CAACI,MAAM,CAAC;;MAEtC;MACA;MACAjB,oBAAW,CAACC,MAAM,CAACmB,KAAK,gEAAyDH,MAAM,EAAG;MAE1F,IAAIE,cAAc,KAAKR,MAAM,EAAE;QAC7B;QACA,IAAI,CAACU,KAAK,EAAE;MACd,CAAC,MAAM,IAAIF,cAAc,KAAKP,YAAY,EAAE;QAC1C;QACA;QACA;QACA,IAAI,CAACjB,WAAW,GAAGC,OAAO;MAC5B;MAEA,IAAI,IAAI,CAACF,aAAa,EAAE;QACtBM,oBAAW,CAACC,MAAM,CAACC,IAAI,uEAC0CiB,cAAc,EAC9E;;QAED;QACA,IAAI,CAACzB,aAAa,CAAC4B,IAAI,CAAC,IAAI,EAAEH,cAAc,EAAEvB,OAAO,CAAC;MACxD;MAEA,IAAI,CAAC2B,SAAS,EAAE;IAClB;;IAEA;AACF;AACA;AACA;EAHE;IAAA;IAAA,OAIA,kBAAS;MACPvB,oBAAW,CAACC,MAAM,CAACC,IAAI,CAAC,oDAAoD,CAAC;MAC7E;MACA,IAAI,CAACV,MAAM,GAAGH,MAAM,CAACG,MAAM,CAACkB,OAAO;MACnC,IAAI,CAACa,SAAS,EAAE;IAClB;;IAEA;AACF;AACA;AACA;AACA;AACA;EALE;IAAA;IAAA,OA/gBA,8BAA4BC,OAAO,EAAEC,QAAQ,EAAE;MAC7C,IAAIC,UAAU,GAAG,IAAI;;MAErB;MACA,IAAIF,OAAO,CAACG,GAAG,GAAGF,QAAQ,CAACG,GAAG,EAAE;QAC9B;QACAF,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACqB,EAAE,WAAQ;MACxC;MACA;MAAA,KACK,IAAIL,OAAO,CAACI,GAAG,GAAGH,QAAQ,CAACE,GAAG,EAAE;QACnC;QACAD,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACsB,EAAE,WAAQ;MACxC;;MAEA;MACA,OAAOJ,UAAU;IACnB;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;EANE;IAAA;IAAA,OAOA,4BAA0BF,OAAO,EAAEC,QAAQ,EAAE;MAC3C,IAAIC,UAAU,GAAG,IAAI;MACrB,IAAMK,kBAAkB,GAAGP,OAAO,CAACQ,MAAM,CAACC,MAAM,KAAK,CAAC;MACtD,IAAMC,mBAAmB,GAAGT,QAAQ,CAACO,MAAM,CAACC,MAAM,KAAK,CAAC;MACxD,IAAME,iBAAiB,GAAGX,OAAO,CAACY,GAAG,GAAGZ,OAAO,CAACG,GAAG;MACnD,IAAMU,kBAAkB,GAAGZ,QAAQ,CAACW,GAAG,GAAGX,QAAQ,CAACE,GAAG;;MAEtD;MACA,IAAII,kBAAkB,IAAIG,mBAAmB,EAAE;QAC7C;QACA,IAAIC,iBAAiB,GAAGE,kBAAkB,EAAE;UAC1C;UACAX,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACqB,EAAE,WAAQ;QACxC;QACA;QAAA,KACK,IAAIM,iBAAiB,GAAGE,kBAAkB,EAAE;UAC/C;UACAX,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACsB,EAAE,WAAQ;QACxC,CAAC,MAAM;UACL;UACA;UACAJ,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAAC8B,EAAE,WAAQ;QACxC;MACF;MAEA,OAAOZ,UAAU;IACnB;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EARE;IAAA;IAAA,OASA,+BAA6BF,OAAO,EAAEC,QAAQ,EAAE;MAC9C,IAAIC,UAAU,GAAG,IAAI;MACrB,IAAMa,eAAe,GAAGf,OAAO,CAACQ,MAAM,CAACC,MAAM,GAAG,CAAC;MACjD,IAAMO,gBAAgB,GAAGf,QAAQ,CAACO,MAAM,CAACC,MAAM,GAAG,CAAC;;MAEnD;MACA,IAAIM,eAAe,IAAI,CAACC,gBAAgB,EAAE;QACxC;QACAd,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACqB,EAAE,WAAQ;MACxC;MACA;MAAA,KACK,IAAI,CAACU,eAAe,IAAIC,gBAAgB,EAAE;QAC7C;QACAd,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACsB,EAAE,WAAQ;MACxC;MAEA,OAAOJ,UAAU;IACnB;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EAPE;IAAA;IAAA,OAQA,0BAAwBF,OAAO,EAAEC,QAAQ,EAAE;MACzC,IAAIC,UAAU,GAAG,IAAI;MACrB,IAAMe,gBAAgB,GAAGjB,OAAO,CAACQ,MAAM,CAAC,CAAC,CAAC;MAC1C,IAAMU,iBAAiB,GAAGjB,QAAQ,CAACO,MAAM,CAAC,CAAC,CAAC;MAE5C,IAAMW,iBAAiB,GAAG,CAACnB,OAAO,CAACoB,KAAK,IAAI,CAACpB,OAAO,CAACY,GAAG;MACxD,IAAMS,kBAAkB,GAAG,CAACpB,QAAQ,CAACmB,KAAK,IAAI,CAACnB,QAAQ,CAACW,GAAG;MAC3D,IAAMU,kBAAkB,GAAGH,iBAAiB,IAAIE,kBAAkB;MAElE,IAAME,cAAc,GAAG,SAAjBA,cAAc,CAAIC,IAAI,EAAErB,GAAG,EAAEC,GAAG;QAAA,OAAKoB,IAAI,CAACC,IAAI,CAAC,UAACC,GAAG;UAAA,OAAKvB,GAAG,GAAGuB,GAAG,IAAIA,GAAG,GAAGtB,GAAG;QAAA,EAAC;MAAA;MACrF;MACA,IAAMuB,kBAAkB,GAAGJ,cAAc,CAACvB,OAAO,CAACQ,MAAM,EAAEP,QAAQ,CAACE,GAAG,EAAEF,QAAQ,CAACG,GAAG,CAAC;MACrF;MACA,IAAMwB,mBAAmB,GAAGL,cAAc,CAACtB,QAAQ,CAACO,MAAM,EAAER,OAAO,CAACG,GAAG,EAAEH,OAAO,CAACI,GAAG,CAAC;MAErF,IAAIkB,kBAAkB,IAAIK,kBAAkB,IAAIC,mBAAmB,EAAE;QACnE;QACA,IAAMC,SAAS,aAAM,CAACP,kBAAkB,cAAI,CAACK,kBAAkB,cAAI,CAACC,mBAAmB,CAAE;;QAEzF;QACA1B,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACG,MAAM,qBAAW0C,SAAS,CAAE;MAC1D,CAAC,MAAM,IAAIZ,gBAAgB,GAAGC,iBAAiB,EAAE;QAC/C;QACAhB,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACqB,EAAE,YAAS;MACzC,CAAC,MAAM;QACL;QACAH,UAAU,aAAMrC,MAAM,CAACmB,IAAI,CAACsB,EAAE,YAAS;MACzC;MAEA,OAAOJ,UAAU;IACnB;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EAPE;IAAA;IAAA,OAQA,iBAAeF,OAAO,EAAEC,QAAQ,EAAE;MAChC,IAAO6B,eAAe,GAAIjE,MAAM,CAAzBiE,eAAe;MACtB,IAA+BzC,OAAO,GAAIxB,MAAM,CAAzCyB,sBAAsB;MAC7B,IAA6ByC,IAAI,GAAIlE,MAAM,CAApCmE,oBAAoB;MAE3B,IAAIF,eAAe,CAAC9B,OAAO,CAAC,IAAI8B,eAAe,CAAC7B,QAAQ,CAAC,EAAE;QACzD,OAAO8B,IAAI,CAAClE,MAAM,CAACmB,IAAI,CAACI,YAAY,EAAE,MAAM,CAAC;MAC/C;MAEA,IAAIa,QAAQ,CAACgC,YAAY,EAAE;QACzB,OAAOF,IAAI,CAAClE,MAAM,CAACqE,YAAY,CAAClC,OAAO,EAAEC,QAAQ,CAAC,EAAE,MAAM,CAAC;MAC7D;MAEA,IAAMR,MAAM,GAAG5B,MAAM,CAACsE,eAAe,CAACnC,OAAO,CAACoC,QAAQ,EAAEnC,QAAQ,CAACmC,QAAQ,CAAC;MAC1E,IAAMtD,MAAM,GAAGjB,MAAM,CAACwE,eAAe,CAAChD,OAAO,CAACI,MAAM,CAAC,CAAC;MAEtD,OAAOsC,IAAI,CAACjD,MAAM,EAAEW,MAAM,CAAC;IAC7B;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EATE;IAAA;IAAA,OAUA,sBAA4BO,OAAO,EAAEC,QAAQ,EAAE;MAC7C,oBAA2CpC,MAAM,CAACmB,IAAI;QAA/CsB,EAAE,iBAAFA,EAAE;QAAED,EAAE,iBAAFA,EAAE;QAAES,EAAE,iBAAFA,EAAE;QAAE3B,MAAM,iBAANA,MAAM;QAAEC,YAAY,iBAAZA,YAAY;MAEvC,IAA+BC,OAAO,GAAIxB,MAAM,CAAzCyB,sBAAsB;MAC7B,IAA6ByC,IAAI,GAAIlE,MAAM,CAApCmE,oBAAoB;MAE3B,IAAMvC,MAAM,GAAG5B,MAAM,CAACsE,eAAe,CAACnC,OAAO,CAACoC,QAAQ,EAAEnC,QAAQ,CAACmC,QAAQ,CAAC;MAC1E,IAAIlC,UAAU,GAAGb,OAAO,CAACI,MAAM,CAAC;MAEhC,IAAIS,UAAU,KAAKI,EAAE,EAAE;QACrB,OAAOyB,IAAI,CAAClE,MAAM,CAACwE,eAAe,CAACnC,UAAU,CAAC,EAAET,MAAM,CAAC;MACzD;MAEAS,UAAU,GAAGrC,MAAM,CAACsE,eAAe,CAACnC,OAAO,CAACoC,QAAQ,EAAEnC,QAAQ,CAACgC,YAAY,CAAC;MAE5E,QAAQ5C,OAAO,CAACa,UAAU,CAAC;QACzB,KAAKG,EAAE;QACP,KAAKS,EAAE;UACLZ,UAAU,GAAGd,YAAY;UACzB;QAEF;UACEc,UAAU,GAAGf,MAAM;MAAC;MAGxB,OAAO4C,IAAI,CAAC7B,UAAU,EAAET,MAAM,CAAC;IACjC;;IAEA;AACF;AACA;AACA;AACA;AACA;EALE;IAAA;IAAA,OAMA,yBAAuBO,OAAO,EAAEC,QAAQ,EAAE;MACxC;MACA;;MAEA,IAAMqC,KAAU,GAAGzE,MAAM,CAAC0E,WAAW,CAACvC,OAAO,CAAC;MAC9C,IAAMwC,KAAU,GAAG3E,MAAM,CAAC0E,WAAW,CAACtC,QAAQ,CAAC;;MAE/C;MACAqC,KAAK,CAAC9B,MAAM,GAAG3C,MAAM,CAAC4E,kBAAkB,CAACH,KAAK,EAAEE,KAAK,CAAC;MACtDA,KAAK,CAAChC,MAAM,GAAG3C,MAAM,CAAC4E,kBAAkB,CAACD,KAAK,EAAEF,KAAK,CAAC;;MAEtD;MACA;MACA,IAAMI,KAAK,GAAG,CACZ7E,MAAM,CAAC8E,oBAAoB,EAC3B9E,MAAM,CAAC+E,kBAAkB,EACzB/E,MAAM,CAACgF,qBAAqB,EAC5BhF,MAAM,CAACiF,gBAAgB,CACxB;MAED,0BAAmBJ,KAAK,4BAAE;QAArB,IAAMK,IAAI;QACb;QACA;QACA,IAAMtD,MAAM,GAAGsD,IAAI,CAACT,KAAK,EAAEE,KAAK,CAAC;QAEjC,IAAI/C,MAAM,EAAE;UACV,OAAOA,MAAM;QACf;MACF;;MAEA;MACA;MACA;MACA,OAAO5B,MAAM,CAACmB,IAAI,CAACgE,KAAK;IAC1B;;IAEA;AACF;AACA;AACA;AACA;EAJE;IAAA;IAAA,OAKA,yBAAuBvD,MAAc,EAAE;MACrC,oBAA+D5B,MAAM,CAACmB,IAAI;QAAnEG,MAAM,iBAANA,MAAM;QAAE2B,EAAE,iBAAFA,EAAE;QAAEkC,KAAK,iBAALA,KAAK;QAAE3C,EAAE,iBAAFA,EAAE;QAAEC,EAAE,iBAAFA,EAAE;QAAE2C,WAAW,iBAAXA,WAAW;QAAE7D,YAAY,iBAAZA,YAAY;MAE3D,IAAIN,MAAM,GAAGkE,KAAK;MAElB,QAAQvD,MAAM;QACZ,KAAKqB,EAAE;QACP,KAAKT,EAAE;UACLvB,MAAM,GAAGmE,WAAW;UACpB;QACF,KAAK3C,EAAE;UACLxB,MAAM,GAAGM,YAAY;UACrB;QACF,KAAKD,MAAM;UACTL,MAAM,GAAGK,MAAM;UACf;QACF;UACEX,oBAAW,CAACC,MAAM,CAACC,IAAI,wDAC2Be,MAAM,sDACvD;MAAC;MAGN,OAAOX,MAAM;IACf;;IAEA;AACF;AACA;AACA;AACA;EAJE;IAAA;IAAA,OAKA,gCAA8BoE,oBAA4B,EAAE;MAC1D,OAAOA,oBAAoB,CAACC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;IAC3C;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;IAEE;AACF;AACA;AACA;AACA;EAJE;IAAA;IAAA,OAKA,qBAAmBf,QAAa,EAAE;MAChC,IAAOgB,OAAO,GAAIhB,QAAQ,CAAnBgB,OAAO;MACd,IAAMC,KAAK,GAAGD,OAAO,CAAC,CAAC,CAAC;MACxB,IAAME,IAAI,GAAGF,OAAO,CAACG,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;;MAEjC;MACA,IAAMnC,KAAK,GAAGgB,QAAQ,CAACoB,UAAU;MACjC,IAAM5C,GAAG,GAAGwB,QAAQ,CAACqB,QAAQ;;MAE7B;MACA,OAAO;QACLrC,KAAK,EAALA,KAAK;QACLR,GAAG,EAAHA,GAAG;QACHyC,KAAK,EAALA,KAAK;QACLC,IAAI,EAAJA,IAAI;QACJ;QACAnD,GAAG,EAAEiB,KAAK,IAAIiC,KAAK;QACnB;QACAjD,GAAG,EAAEkD,IAAI,IAAI1C,GAAG;QAChB;QACAwC,OAAO,EAAPA;MACF,CAAC;IACH;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;EANE;IAAA;IAAA,OAOA,4BAA0BM,QAAa,EAAEC,SAAc,EAAE;MACvD,IAAMC,IAAS,GAAG,0BAAWF,QAAQ,CAACN,OAAO,EAAEO,SAAS,CAACP,OAAO,CAAC;MAEjE,IAAOhC,KAAK,GAASuC,SAAS,CAAvBvC,KAAK;QAAER,GAAG,GAAI+C,SAAS,CAAhB/C,GAAG;MAEjB,OAAO/C,MAAM,CAACgG,oBAAoB,CAACD,IAAI,EAAExC,KAAK,EAAER,GAAG,CAAC;IACtD;;IAEA;AACF;AACA;AACA;AACA;AACA;AACA;EANE;IAAA;IAAA,OAOA,8BAA4BY,IAAmB,EAAEgC,UAAkB,EAAEC,QAAgB,EAAE;MACrF;MACA,IAAMK,MAAM,GAAGtC,IAAI,CAACuC,MAAM,CAAC,UAACC,GAAG;QAAA,OAAKA,GAAG,GAAGR,UAAU,IAAIQ,GAAG,GAAGP,QAAQ;MAAA,EAAC;;MAEvE;MACA,OAAOK,MAAM,CAACG,IAAI,CAAC,UAACC,CAAC,EAAEC,CAAC;QAAA,OAAKD,CAAC,GAAGC,CAAC;MAAA,EAAC;IACrC;EAAC;IAAA;IAAA,OAsCD,yBAAuBpF,KAAK,EAAE;MAAA;MAC5B,IAAOqD,QAAQ,GAAIrD,KAAK,CAAjBqD,QAAQ;MACf,IAAMgC,eAAe,GAAG,uBAAChC,QAAQ,CAACgB,OAAO,8CAAhB,kBAAkB3C,MAAM;MACjD,IAAM4D,aAAa,GAAGjC,QAAQ,CAACoB,UAAU,KAAK,CAAC,IAAIpB,QAAQ,CAACqB,QAAQ,KAAK,CAAC;MAE1E,OAAOW,eAAe,IAAIC,aAAa;IACzC;;IAEA;AACF;AACA;AACA;AACA;AACA;EALE;IAAA;IAAA,OAMA,gBAAcrF,IAAI,EAAE;MAClB,IAAI,CAACA,IAAI,IAAI,CAACA,IAAI,CAACoD,QAAQ,EAAE;QAC3B,OAAO,KAAK;MACd;MACA,IAAMkC,OAAO,GAAG,SAAVA,OAAO,CAAIC,IAAI;QAAA,OAAKC,MAAM,CAACC,SAAS,CAACC,cAAc,CAAC5E,IAAI,CAACd,IAAI,CAACoD,QAAQ,EAAEmC,IAAI,CAAC;MAAA;MAEnF,IAAID,OAAO,CAAC,YAAY,CAAC,IAAIA,OAAO,CAAC,UAAU,CAAC,EAAE;QAChD,OAAO,IAAI;MACb;MAEA,OAAO,KAAK;IACd;EAAC;IAAA;IAAA,OA4DD,8BAA4BK,OAAe,EAAEC,OAAe,EAAE;MAC5D,iBAAUD,OAAO,cAAIC,OAAO;IAC9B;EAAC;IAAA;IAAA,OAyED,yBAAuBC,SAAiB,EAAE3E,UAAkB,EAAE;MAC5D;MACA,IAAM4E,IAAI,GAAG,SAAPA,IAAI,CAAIC,OAAO;QAAA,OAAKA,OAAO,CAACC,IAAI,CAAC,EAAE,CAAC,CAACC,OAAO,CAAC,SAAS,EAAE,GAAG,CAAC;MAAA;MAElE,IAAMC,aAAa,GAAG;QACpBpE,EAAE,YAAKjD,MAAM,CAACmB,IAAI,CAACsB,EAAE,sCAAmC;QACxDA,EAAE,YAAKzC,MAAM,CAACmB,IAAI,CAACsB,EAAE,oCAAiC;QACtDD,EAAE,YAAKxC,MAAM,CAACmB,IAAI,CAACqB,EAAE;MACvB,CAAC;MAED,IAAM8E,QAAQ,GAAG;QACfC,KAAK,EAAE;UACLC,KAAK,EAAE,0BAA0B;UACjCC,WAAW,EAAER,IAAI,6RACuB;UACxCS,KAAK,EAAE;QACT,CAAC;QAEDC,KAAK,EAAE;UACLH,KAAK,EAAE,0BAA0B;UACjCC,WAAW,EAAER,IAAI,+QACa;UAC9BS,KAAK,EAAE;QACT,CAAC;QAEDE,KAAK,EAAE;UACLJ,KAAK,EAAE,wBAAwB;UAC/BC,WAAW,EAAER,IAAI,qUACuC;UACxDS,KAAK,EAAE;QACT,CAAC;QAEDG,KAAK,EAAE;UACLL,KAAK,EAAE,wBAAwB;UAC/BC,WAAW,EAAER,IAAI,yTACiC;UAClDS,KAAK,EAAE;QACT,CAAC;QAEDI,KAAK,EAAE;UACLN,KAAK,EAAE,wBAAwB;UAC/BC,WAAW,EAAER,IAAI,+ZAEO;UACxBS,KAAK,EAAE;QACT,CAAC;QAEDK,KAAK,EAAE;UACLP,KAAK,EAAE,2BAA2B;UAClCC,WAAW,EAAER,IAAI,2cAE2C;UAC5DS,KAAK,EAAE;QACT,CAAC;QAEDM,KAAK,EAAE;UACLR,KAAK,EAAE,2BAA2B;UAClCC,WAAW,EAAER,IAAI,+bAE2C;UAC5DS,KAAK,EAAE;QACT,CAAC;QAEDO,MAAM,EAAE;UACNT,KAAK,EAAE,sBAAsB;UAC7BC,WAAW,EAAER,IAAI,+jBAGsB;UACvCS,KAAK,EAAE;QACT,CAAC;QAEDQ,MAAM,EAAE;UACNV,KAAK,EAAE,sBAAsB;UAC7BC,WAAW,EAAER,IAAI,qdAEiC;UAClDS,KAAK,EAAE;QACT,CAAC;QAEDS,MAAM,EAAE;UACNX,KAAK,EAAE,sBAAsB;UAC7BC,WAAW,EAAER,IAAI,yPACI;UACrBS,KAAK,EAAE;QACT;MACF,CAAC;MAED,IAAMU,QAAQ,GAAGd,QAAQ,CAACN,SAAS,CAAC;MAEpCoB,QAAQ,CAACZ,KAAK,oBAAaY,QAAQ,CAACZ,KAAK,CAAE;MAC3CY,QAAQ,CAACC,UAAU,GAAGhB,aAAa,CAAChF,UAAU,CAAC;MAE/C,OAAO+F,QAAQ;IACjB;EAAC;EAAA;AAAA;AAAA;AAAA,8BA5pBkBpI,MAAM,YAET;EACdI,IAAI,EAAE,MAAM;EACZU,MAAM,EAAE,QAAQ;EAChBO,OAAO,EAAE;AACX,CAAC;AAAA,8BANkBrB,MAAM,UASX;EACZiD,EAAE,EAAE,OAAO;EACXT,EAAE,EAAE,cAAc;EAClBC,EAAE,EAAE,WAAW;EACfnB,MAAM,EAAE,QAAQ;EAChBC,YAAY,EAAE,cAAc;EAC5B6D,WAAW,EAAE,aAAa;EAC1BD,KAAK,EAAE;AACT,CAAC"}
1
+ {"version":3,"names":["MAX_OOO_DELTA_COUNT","OOO_DELTA_WAIT_TIME","OOO_DELTA_WAIT_TIME_RANDOM_DELAY","Parser","deltaCompareFunc","left","right","loci","LT","GT","extract","extractComparisonState","isSequenceEmpty","result","compareSequence","baseSequence","queue","SortedQueue","status","onDeltaAction","workingCopy","syncTimer","incomingFullDto","isLoci","LoggerProxy","logger","info","comparisonResult","compareFullDtoSequence","USE_INCOMING","newLoci","isValid","size","processDeltaEvent","action","locus","enqueue","triggerSync","reason","stopSyncTimer","pause","DESYNC","timeout","Math","random","setTimeout","clearTimeout","WAIT","dequeue","isValidLocus","nextEvent","compare","lociComparison","debug","needToWait","startSyncTimer","locus2string","sequence","entries","length","at","current","incoming","comparison","min","max","currentIsNotUnique","unique","incomingIsNotUnique","currentTotalRange","end","incomingTotalRange","EQ","currentIsUnique","incomingIsUnique","currentUniqueMin","incomingUniqueMin","currentHasNoRange","start","incomingHasNoRange","neitherSeqHasRange","hasUniqOverlap","list","some","seq","currentUniqOverlap","incomingUniqOverlap","debugInfo","pack","packComparisonResult","compareDelta","compareToAction","Metrics","sendBehavioralMetric","BEHAVIORAL_METRICS","LOCUS_DELTA_OUT_OF_ORDER","stack","Error","slice","USE_CURRENT","local","getMetaData","delta","getUniqueSequences","rules","checkSequenceOverlap","checkUnequalRanges","checkForUniqueEntries","checkIfOutOfSync","rule","ERROR","lociComparisonResult","split","first","last","rangeStart","rangeEnd","baseLoci","otherLoci","diff","getNumbersOutOfRange","output","filter","num","sort","a","b","hasEmptyEntries","hasEmptyRange","hasProp","prop","Object","prototype","hasOwnProperty","call","newData","oldData","debugCode","mStr","strings","join","replace","resolutionMap","debugMap","SO001","title","description","logic","SO002","UR001","UR002","UR003","UE001","UE002","OOS001","OOS002","OOS003","debugObj","resolution"],"sources":["parser.ts"],"sourcesContent":["import {difference} from 'lodash';\n\nimport SortedQueue from '../common/queue';\nimport LoggerProxy from '../common/logs/logger-proxy';\n\nimport Metrics from '../metrics';\nimport BEHAVIORAL_METRICS from '../metrics/constants';\n\nconst MAX_OOO_DELTA_COUNT = 5; // when we receive an out-of-order delta and the queue builds up to MAX_OOO_DELTA_COUNT, we do a sync with Locus\nconst OOO_DELTA_WAIT_TIME = 10000; // [ms] minimum wait time before we do a sync if we get out-of-order deltas\nconst OOO_DELTA_WAIT_TIME_RANDOM_DELAY = 5000; // [ms] max random delay added to OOO_DELTA_WAIT_TIME\n\ntype LocusDeltaDto = {\n baseSequence: {\n rangeStart: number;\n rangeEnd: number;\n entries: number[];\n };\n sequence: {\n rangeStart: number;\n rangeEnd: number;\n entries: number[];\n };\n syncUrl: string;\n};\n\n/**\n * Locus Delta Parser\n * @private\n * https://sqbu-github.cisco.com/WebExSquared/cloud-apps/wiki/Locus-Delta-Events\n */\nexport default class Parser {\n // processing status\n status:\n | 'IDLE' // not doing anything\n | 'PAUSED' // paused, because we are doing a sync\n | 'WORKING' // processing a delta event\n | 'BLOCKED'; // received an out-of-order delta, so waiting for the missing one\n\n // loci comparison states\n static loci = {\n EQ: 'EQUAL',\n GT: 'GREATER_THAN',\n LT: 'LESS_THAN',\n DESYNC: 'DESYNC',\n USE_INCOMING: 'USE_INCOMING',\n USE_CURRENT: 'USE_CURRENT',\n WAIT: 'WAIT',\n ERROR: 'ERROR',\n };\n\n queue: SortedQueue<LocusDeltaDto>;\n workingCopy: any;\n syncTimer: null | number | NodeJS.Timeout;\n\n /**\n * @constructs Parser\n */\n constructor() {\n const deltaCompareFunc = (left: LocusDeltaDto, right: LocusDeltaDto) => {\n const {LT, GT} = Parser.loci;\n const {extractComparisonState: extract} = Parser;\n\n if (Parser.isSequenceEmpty(left)) {\n return -1;\n }\n if (Parser.isSequenceEmpty(right)) {\n return 1;\n }\n const result = extract(Parser.compareSequence(left.baseSequence, right.baseSequence));\n\n if (result === LT) {\n return -1;\n }\n if (result === GT) {\n return 1;\n }\n\n return 0;\n };\n\n this.queue = new SortedQueue<LocusDeltaDto>(deltaCompareFunc);\n this.status = 'IDLE';\n this.onDeltaAction = null;\n this.workingCopy = null;\n this.syncTimer = null;\n }\n\n /**\n * Returns a debug string representing a locus delta - useful for logging\n *\n * @param {LocusDeltaDto} locus Locus delta\n * @returns {string}\n */\n static locus2string(locus: LocusDeltaDto) {\n if (!locus.sequence?.entries) {\n return 'invalid';\n }\n\n return locus.sequence.entries.length ? `seq=${locus.sequence.entries.at(-1)}` : 'empty';\n }\n\n /**\n * Checks if two sequences overlap in time,\n * the sequence with the higher minimum value is greater.\n * Chooses sequence with most recent data.\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {string} loci comparison state\n */\n static checkSequenceOverlap(current, incoming) {\n let comparison = null;\n\n // if earliest working copy sequence is more recent than last incoming sequence\n if (current.min > incoming.max) {\n // choose left side (current)\n comparison = `${Parser.loci.GT}:SO001`;\n }\n // if last working copy sequence is before the earliest incoming sequence\n else if (current.max < incoming.min) {\n // choose right side (incoming)\n comparison = `${Parser.loci.LT}:SO002`;\n }\n\n // if no match above, defaults to null\n return comparison;\n }\n\n /**\n * Checks if two sequences have unequal ranges.\n * Chooses sequence with most larger range.\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {object} loci comparison\n */\n static checkUnequalRanges(current, incoming) {\n let comparison = null;\n const currentIsNotUnique = current.unique.length === 0;\n const incomingIsNotUnique = incoming.unique.length === 0;\n const currentTotalRange = current.end - current.min;\n const incomingTotalRange = incoming.end - incoming.min;\n\n // no unique values for both loci\n if (currentIsNotUnique && incomingIsNotUnique) {\n // current working copy loci has a larger range\n if (currentTotalRange > incomingTotalRange) {\n // choose left side (current)\n comparison = `${Parser.loci.GT}:UR001`;\n }\n // incoming delta loci has a larger range\n else if (currentTotalRange < incomingTotalRange) {\n // choose right side (incoming)\n comparison = `${Parser.loci.LT}:UR002`;\n } else {\n // with no unique entries and with ranges either absent or\n // of the same size, the sequences are considered equal.\n comparison = `${Parser.loci.EQ}:UR003`;\n }\n }\n\n return comparison;\n }\n\n /**\n * Checks if either sequences has unique entries.\n * Entries are considered unique if they do not overlap\n * with other Loci sequences or range values.\n * Chooses sequence with the unique entries.\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {string} loci comparison state\n */\n static checkForUniqueEntries(current, incoming) {\n let comparison = null;\n const currentIsUnique = current.unique.length > 0;\n const incomingIsUnique = incoming.unique.length > 0;\n\n // current has unique entries and incoming does not\n if (currentIsUnique && !incomingIsUnique) {\n // choose left side (current)\n comparison = `${Parser.loci.GT}:UE001`;\n }\n // current has no unique entries but incoming does\n else if (!currentIsUnique && incomingIsUnique) {\n // choose right side (incoming)\n comparison = `${Parser.loci.LT}:UE002`;\n }\n\n return comparison;\n }\n\n /**\n * Checks both Locus Delta objects to see if they are\n * out of sync with one another. If so sends a DESYNC\n * request to the server.\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {string} loci comparison state\n */\n static checkIfOutOfSync(current, incoming) {\n let comparison = null;\n const currentUniqueMin = current.unique[0];\n const incomingUniqueMin = incoming.unique[0];\n\n const currentHasNoRange = !current.start && !current.end;\n const incomingHasNoRange = !incoming.start && !incoming.end;\n const neitherSeqHasRange = currentHasNoRange && incomingHasNoRange;\n\n const hasUniqOverlap = (list, min, max) => list.some((seq) => min < seq && seq < max);\n // current unique entries overlap the total range of incoming\n const currentUniqOverlap = hasUniqOverlap(current.unique, incoming.min, incoming.max);\n // vice-versa, incoming unique entries overlap the total range of current\n const incomingUniqOverlap = hasUniqOverlap(incoming.unique, current.min, current.max);\n\n if (neitherSeqHasRange || currentUniqOverlap || incomingUniqOverlap) {\n // outputs string indicating which condition occurred. ex: 0,1,0\n const debugInfo = `${+neitherSeqHasRange},${+currentUniqOverlap},${+incomingUniqOverlap}`;\n\n // send DESYNC to server\n comparison = `${Parser.loci.DESYNC}:OOS001:${debugInfo}`;\n } else if (currentUniqueMin > incomingUniqueMin) {\n // choose left side (current)\n comparison = `${Parser.loci.GT}:OOS002`;\n } else {\n // choose right side (incoming)\n comparison = `${Parser.loci.LT}:OOS003`;\n }\n\n return comparison;\n }\n\n /**\n * Compares two loci to determine which one contains the most recent state\n * @instance\n * @memberof Locus\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @returns {string} loci comparison state\n */\n static compare(current, incoming) {\n const {isSequenceEmpty} = Parser;\n const {extractComparisonState: extract} = Parser;\n const {packComparisonResult: pack} = Parser;\n\n if (isSequenceEmpty(current) || isSequenceEmpty(incoming)) {\n return pack(Parser.loci.USE_INCOMING, 'C001');\n }\n\n if (incoming.baseSequence) {\n return pack(Parser.compareDelta(current, incoming), 'C002');\n }\n\n const result = Parser.compareSequence(current.sequence, incoming.sequence);\n const action = Parser.compareToAction(extract(result));\n\n return pack(action, result);\n }\n\n /**\n * Compares two loci sequences (with delta params) and indicates what action\n * to take.\n * @instance\n * @memberof Locus\n * @param {Types~Locus} current\n * @param {Types~Locus} incoming\n * @private\n * @returns {string} loci comparison state\n */\n private static compareDelta(current, incoming) {\n const {LT, GT, EQ, DESYNC, USE_INCOMING, WAIT} = Parser.loci;\n\n const {extractComparisonState: extract} = Parser;\n const {packComparisonResult: pack} = Parser;\n\n const result = Parser.compareSequence(current.sequence, incoming.sequence);\n let comparison = extract(result);\n\n if (comparison !== LT) {\n return pack(Parser.compareToAction(comparison), result);\n }\n\n comparison = Parser.compareSequence(current.sequence, incoming.baseSequence);\n\n switch (extract(comparison)) {\n case GT:\n case EQ:\n comparison = USE_INCOMING;\n break;\n\n case LT:\n if (extract(Parser.compareSequence(incoming.baseSequence, incoming.sequence)) === EQ) {\n // special case where Locus sends a delta with baseSequence === sequence to trigger a sync,\n // because the delta event is too large to be sent over mercury connection\n comparison = DESYNC;\n } else {\n // the incoming locus has baseSequence from the future, so it is out-of-order,\n // we are missing 1 or more locus that should be in front of it, we need to wait for it\n comparison = WAIT;\n\n Metrics.sendBehavioralMetric(BEHAVIORAL_METRICS.LOCUS_DELTA_OUT_OF_ORDER, {\n stack: new Error().stack,\n });\n }\n break;\n default:\n comparison = DESYNC;\n }\n\n return pack(comparison, result);\n }\n\n /**\n * Compares Locus sequences - it should be called only for full Locus DTOs, not deltas\n *\n * @param {Types~Locus} current Current working copy\n * @param {Types~Locus} incomingFullDto New Full Locus DTO\n * @returns {string} either Parser.loci.USE_INCOMING or Parser.loci.USE_CURRENT\n */\n static compareFullDtoSequence(current, incomingFullDto) {\n if (Parser.isSequenceEmpty(current) || Parser.isSequenceEmpty(incomingFullDto)) {\n return Parser.loci.USE_INCOMING;\n }\n\n // the sequence.entries list will always contain at least 1 entry\n // https://sqbu-github.cisco.com/WebExSquared/cloud-apps/wiki/Locus-Sequence-Comparison-Algorithm\n\n return incomingFullDto.sequence.entries.slice(-1)[0] > current.sequence.entries.slice(-1)[0]\n ? Parser.loci.USE_INCOMING\n : Parser.loci.USE_CURRENT;\n }\n\n /**\n * Returns true if the incoming full locus DTO is newer than the current working copy\n *\n * @param {Types~Locus} incomingFullDto New Full Locus DTO\n * @returns {string} either Parser.loci.USE_INCOMING or Parser.loci.USE_CURRENT\n */\n isNewFullLocus(incomingFullDto) {\n if (!Parser.isLoci(incomingFullDto)) {\n LoggerProxy.logger.info('Locus-info:parser#isNewFullLocus --> Ignoring non-locus object.');\n\n return false;\n }\n\n if (!this.workingCopy) {\n // we don't have a working copy yet, so any full locus is better than nothing\n return true;\n }\n\n const comparisonResult = Parser.compareFullDtoSequence(this.workingCopy, incomingFullDto);\n\n return comparisonResult === Parser.loci.USE_INCOMING;\n }\n\n /**\n * Compares Locus sequences\n * @param {Types~Locus} current Current working copy\n * @param {Types~Locus} incoming New Locus delta\n * @returns {string}\n */\n static compareSequence(current, incoming) {\n // Locus sequence comparison rules in order of priority.\n // https://sqbu-github.cisco.com/WebExSquared/cloud-apps/wiki/Locus-Sequence-Comparison-Algorithm\n\n const local: any = Parser.getMetaData(current);\n const delta: any = Parser.getMetaData(incoming);\n\n // update loci metadata\n local.unique = Parser.getUniqueSequences(local, delta);\n delta.unique = Parser.getUniqueSequences(delta, local);\n\n // Locus sequence comparison rules\n // order matters\n const rules = [\n Parser.checkSequenceOverlap,\n Parser.checkUnequalRanges,\n Parser.checkForUniqueEntries,\n Parser.checkIfOutOfSync,\n ];\n\n for (const rule of rules) {\n // Rule only returns a value if the rule applies,\n // otherwise returns null.\n const result = rule(local, delta);\n\n if (result) {\n return result;\n }\n }\n\n // error, none of rules above applied\n // should never get here as last rule\n // should be catch all.\n return Parser.loci.ERROR;\n }\n\n /**\n * Transates the result of a sequence comparison into an intended behavior\n * @param {string} result\n * @returns {string} Locus comparison action\n */\n static compareToAction(result: string) {\n const {DESYNC, EQ, ERROR, GT, LT, USE_CURRENT, USE_INCOMING} = Parser.loci;\n\n let action = ERROR;\n\n switch (result) {\n case EQ:\n case GT:\n action = USE_CURRENT;\n break;\n case LT:\n action = USE_INCOMING;\n break;\n case DESYNC:\n action = DESYNC;\n break;\n default:\n LoggerProxy.logger.info(\n `Locus-info:parser#compareToAction --> Error: ${result} is not a recognized sequence comparison result.`\n );\n }\n\n return action;\n }\n\n /**\n * Extracts a loci comparison from a string of data.\n * @param {string} lociComparisonResult Comparison result with extra data\n * @returns {string} Comparison of EQ, LT, GT, or DESYNC.\n */\n static extractComparisonState(lociComparisonResult: string) {\n return lociComparisonResult.split(':')[0];\n }\n\n /**\n * @typedef {object} LociMetadata\n * @property {number} start - Starting sequence number\n * @property {number} end - Ending sequence number\n * @property {number} first - First sequence number\n * @property {number} last - Last sequence number\n * @property {number} min - Minimum sequence number\n * @property {number} max - Maximum sequence number\n * @property {number} entries - Loci sequence entries\n */\n\n /**\n * Metadata for Locus delta\n * @param {Array.<number>} sequence Locus delta sequence\n * @returns {LociMetadata} Locus Delta Metadata\n */\n static getMetaData(sequence: any) {\n const {entries} = sequence;\n const first = entries[0];\n const last = entries.slice(-1)[0];\n\n // rangeStart or rangeEnd is 0 if a range doesn't exist\n const start = sequence.rangeStart;\n const end = sequence.rangeEnd;\n\n // sequence data\n return {\n start,\n end,\n first,\n last,\n // Rule is: rangeStart <= rangeEnd <= min(entries)\n min: start || first,\n // Grab last entry if exist else default to rangeEnd\n max: last || end,\n // keep reference to actual sequence entries\n entries,\n };\n }\n\n /**\n * Compares two Locus delta objects and notes unique\n * values contained within baseLoci.\n * @param {LociMetadata} baseLoci\n * @param {LociMetadata} otherLoci\n * @returns {Array.<number>} List of unique sequences\n */\n static getUniqueSequences(baseLoci: any, otherLoci: any) {\n const diff: any = difference(baseLoci.entries, otherLoci.entries);\n\n const {start, end} = otherLoci;\n\n return Parser.getNumbersOutOfRange(diff, start, end);\n }\n\n /**\n * Returns an array of numbers outside of a given range.\n * @param {Array.<number>} list Array to filter\n * @param {number} rangeStart Start of range\n * @param {number} rangeEnd End of range\n * @returns {Array.<number>} Array of numbers sorted ASC\n */\n static getNumbersOutOfRange(list: Array<number>, rangeStart: number, rangeEnd: number) {\n // Collect all numbers if number is outside of specified range\n const output = list.filter((num) => num < rangeStart || num > rangeEnd);\n\n // sort ascending\n return output.sort((a, b) => a - b);\n }\n\n /**\n * Checks if newLoci or workingCopy is invalid.\n * @param {Types~Locus} newLoci\n * @returns {boolean}\n */\n isValidLocus(newLoci) {\n let isValid = false;\n const {isLoci} = Parser;\n\n // one or both objects are not locus delta events\n if (!isLoci(this.workingCopy) || !isLoci(newLoci)) {\n LoggerProxy.logger.info(\n 'Locus-info:parser#processDeltaEvent --> Ignoring non-locus object. workingCopy:',\n this.workingCopy,\n 'newLoci:',\n newLoci\n );\n } else {\n isValid = true;\n }\n\n return isValid;\n }\n\n /**\n * Determines if a paricular locus's sequence is empty\n * @param {Types~Locus} locus\n * @returns {bool}\n */\n static isSequenceEmpty(locus) {\n const {sequence} = locus;\n const hasEmptyEntries = !sequence.entries?.length;\n const hasEmptyRange = sequence.rangeStart === 0 && sequence.rangeEnd === 0;\n\n return hasEmptyEntries && hasEmptyRange;\n }\n\n /**\n * Determines if an object has basic\n * structure of a locus object.\n * @param {Types~Locus} loci\n * @returns {boolean}\n */\n static isLoci(loci) {\n if (!loci || !loci.sequence) {\n return false;\n }\n const hasProp = (prop) => Object.prototype.hasOwnProperty.call(loci.sequence, prop);\n\n if (hasProp('rangeStart') && hasProp('rangeEnd')) {\n return true;\n }\n\n return false;\n }\n\n /**\n * Processes next event in queue,\n * if queue is empty sets status to idle.\n * @returns {undefined}\n */\n nextEvent() {\n if (this.status === 'PAUSED') {\n LoggerProxy.logger.info('Locus-info:parser#nextEvent --> Locus parser paused.');\n\n return;\n }\n\n if (this.status === 'BLOCKED') {\n LoggerProxy.logger.info(\n 'Locus-info:parser#nextEvent --> Locus parser blocked by out-of-order delta.'\n );\n\n return;\n }\n\n // continue processing until queue is empty\n if (this.queue.size() > 0) {\n this.processDeltaEvent();\n } else {\n this.status = 'IDLE';\n }\n }\n\n /**\n * Function handler for delta actions,\n * is set by instance callee.\n * @param {string} action Locus delta action\n * @param {Types~Locus} locus Locus delta\n * @returns {undefined}\n */\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n onDeltaAction(action: string, locus) {}\n\n /**\n * Event handler for locus delta events\n * @param {Types~Locus} loci Locus Delta\n * @returns {undefined}\n */\n onDeltaEvent(loci) {\n // enqueue the new loci\n this.queue.enqueue(loci);\n\n if (this.onDeltaAction) {\n if (this.status === 'BLOCKED') {\n if (this.queue.size() > MAX_OOO_DELTA_COUNT) {\n this.triggerSync('queue too big, blocked on out-of-order delta');\n } else {\n this.processDeltaEvent();\n }\n } else if (this.status === 'IDLE') {\n // Update status, ensure we only process one event at a time.\n this.status = 'WORKING';\n\n this.processDeltaEvent();\n }\n }\n }\n\n /**\n * Appends new data onto a string of existing data.\n * @param {string} newData\n * @param {string} oldData\n * @returns {string}\n */\n static packComparisonResult(newData: string, oldData: string) {\n return `${newData}:${oldData}`;\n }\n\n /**\n * Pause locus processing\n * @returns {undefined}\n */\n pause() {\n this.status = 'PAUSED';\n LoggerProxy.logger.info('Locus-info:parser#pause --> Locus parser paused.');\n }\n\n /**\n * Triggers a sync with Locus\n *\n * @param {string} reason used just for logging\n * @returns {undefined}\n */\n private triggerSync(reason: string) {\n LoggerProxy.logger.info(`Locus-info:parser#triggerSync --> doing sync, reason: ${reason}`);\n this.stopSyncTimer();\n this.pause();\n this.onDeltaAction(Parser.loci.DESYNC, this.workingCopy);\n }\n\n /**\n * Starts a timer with a random delay. When that timer expires we will do a sync.\n *\n * The main purpose of this timer is to handle a case when we get some out-of-order deltas,\n * so we start waiting to receive the missing delta. If that delta never arrives, this timer\n * will trigger a sync with Locus.\n *\n * @returns {undefined}\n */\n private startSyncTimer() {\n if (this.syncTimer === null) {\n const timeout = OOO_DELTA_WAIT_TIME + Math.random() * OOO_DELTA_WAIT_TIME_RANDOM_DELAY;\n\n this.syncTimer = setTimeout(() => {\n this.syncTimer = null;\n this.triggerSync('timer expired, blocked on out-of-order delta');\n }, timeout);\n }\n }\n\n /**\n * Stops the timer for triggering a sync\n *\n * @returns {undefined}\n */\n private stopSyncTimer() {\n if (this.syncTimer !== null) {\n clearTimeout(this.syncTimer);\n this.syncTimer = null;\n }\n }\n\n /**\n * Processes next locus delta in the queue,\n * continues until the queue is empty\n * or cleared.\n * @returns {undefined}\n */\n processDeltaEvent() {\n const {DESYNC, USE_INCOMING, WAIT} = Parser.loci;\n const {extractComparisonState: extract} = Parser;\n const newLoci = this.queue.dequeue();\n\n if (!this.isValidLocus(newLoci)) {\n this.nextEvent();\n\n return;\n }\n\n const result = Parser.compare(this.workingCopy, newLoci);\n const lociComparison = extract(result);\n\n // limited debugging, use chrome extension\n // for full debugging.\n LoggerProxy.logger.debug(`Locus-info:parser#processDeltaEvent --> Locus Debug: ${result}`);\n\n let needToWait = false;\n\n if (lociComparison === DESYNC) {\n // wait for desync response\n this.pause();\n } else if (lociComparison === USE_INCOMING) {\n // update working copy for future comparisons.\n // Note: The working copy of parser gets updated in .onFullLocus()\n // and here when USE_INCOMING locus.\n this.workingCopy = newLoci;\n } else if (lociComparison === WAIT) {\n // we've taken newLoci from the front of the queue, so put it back there as we have to wait\n // for the one that should be in front of it, before we can process it\n this.queue.enqueue(newLoci);\n needToWait = true;\n }\n\n if (needToWait) {\n this.status = 'BLOCKED';\n this.startSyncTimer();\n } else {\n this.stopSyncTimer();\n\n if (this.status === 'BLOCKED') {\n // we are not blocked anymore\n this.status = 'WORKING';\n\n LoggerProxy.logger.info(\n `Locus-info:parser#processDeltaEvent --> received delta that we were waiting for ${Parser.locus2string(\n newLoci\n )}, not blocked anymore`\n );\n }\n }\n\n if (this.onDeltaAction) {\n LoggerProxy.logger.info(\n `Locus-info:parser#processDeltaEvent --> Locus Delta ${Parser.locus2string(\n newLoci\n )}, Action: ${lociComparison}`\n );\n\n this.onDeltaAction(lociComparison, newLoci);\n }\n\n this.nextEvent();\n }\n\n /**\n * Resume from a paused state\n * @returns {undefined}\n */\n resume() {\n LoggerProxy.logger.info('Locus-info:parser#resume --> Locus parser resumed.');\n this.status = 'WORKING';\n this.nextEvent();\n }\n\n /**\n * Gets related debug info for given error code\n * @param {string} debugCode Debug code\n * @param {string} comparison Locus comparison string\n * @returns {object} Debug message\n */\n static getDebugMessage(debugCode: string, comparison: string) {\n // removes extra spaces from multiline string\n const mStr = (strings) => strings.join('').replace(/\\s{2,}/g, ' ');\n\n const resolutionMap = {\n EQ: `${Parser.loci.LT}: is equal (current == incoming).`,\n LT: `${Parser.loci.LT}: choose right side (incoming).`,\n GT: `${Parser.loci.GT}: choose left side (current).`,\n };\n\n const debugMap = {\n SO001: {\n title: 'checkSequenceOverlap-001',\n description: mStr`Occurs if earliest working copy sequence is more \\\n recent than last incoming sequence.`,\n logic: 'current.min > incoming.max',\n },\n\n SO002: {\n title: 'checkSequenceOverlap-002',\n description: mStr`Occurs if last working copy sequence is before the \\\n earliest incoming sequence.`,\n logic: 'current.max < incoming.min',\n },\n\n UR001: {\n title: 'checkUnequalRanges-001',\n description: mStr`Occurs if there are no unique values for both loci, \\\n and the current working copy loci has a larger range.`,\n logic: 'currentTotalRange > incomingTotalRange',\n },\n\n UR002: {\n title: 'checkUnequalRanges-002',\n description: mStr`Occurs if there are no unique values for both loci, \\\n and the incoming delta loci has a larger range.`,\n logic: 'currentTotalRange < incomingTotalRange',\n },\n\n UR003: {\n title: 'checkUnequalRanges-003',\n description: mStr`Occurs if there are no unique values for both loci, \\\n and with ranges either absent or of the same size, the sequences \\\n are considered equal.`,\n logic: 'currentTotalRange == incomingTotalRange',\n },\n\n UE001: {\n title: 'checkForUniqueEntries-001',\n description: mStr`Occurs if current loci has unique entries and \\\n incoming does not. Entries are considered unique if they \\\n do not overlap with other Loci sequences or range values.`,\n logic: 'currentIsUnique && !incomingIsUnique',\n },\n\n UE002: {\n title: 'checkForUniqueEntries-002',\n description: mStr`Occurs if current has no unique entries but \\\n incoming does. Entries are considered unique if they \\\n do not overlap with other Loci sequences or range values.`,\n logic: '!currentIsUnique && incomingIsUnique',\n },\n\n OOS001: {\n title: 'checkIfOutOfSync-001',\n description: mStr`Occurs if neither sequence has a range, or \\\n if the current loci unique entries overlap the total range of the \\\n incoming sequence, or if the incoming unique entries overlap \\\n the total range of current sequence.`,\n logic: 'neitherSeqHasRange || currentUniqOverlap || incomingUniqOverlap',\n },\n\n OOS002: {\n title: 'checkIfOutOfSync-002',\n description: mStr`Occurs if the minimum value from sequences that are \\\n unique to the current loci is greater than the minimum value from \\\n sequences that are unique to the incoming loci.`,\n logic: 'currentUniqueMin > incomingUniqueMin',\n },\n\n OOS003: {\n title: 'checkIfOutOfSync-003',\n description: mStr`Occurs if none of the comparison rules applied. \\\n It is a catch all.`,\n logic: 'else (catch all)',\n },\n };\n\n const debugObj = debugMap[debugCode];\n\n debugObj.title = `Debug: ${debugObj.title}`;\n debugObj.resolution = resolutionMap[comparison];\n\n return debugObj;\n }\n}\n"],"mappings":"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"}