fmode-ng 0.0.113 → 0.0.115

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 (275) hide show
  1. package/LICENSE.md +8 -0
  2. package/README.md +37 -37
  3. package/esm2022/fmode-ng.mjs +10 -5
  4. package/esm2022/lib/aigc/agent/fm-agent-task/fm-agent-task.component.mjs +8 -92
  5. package/esm2022/lib/aigc/agent/index.mjs +10 -3
  6. package/esm2022/lib/aigc/avatar/avatar.module.mjs +10 -45
  7. package/esm2022/lib/aigc/avatar/comp-avatar-particle/avatar.role.mjs +10 -2
  8. package/esm2022/lib/aigc/avatar/comp-avatar-particle/comp-avatar-particle.component.mjs +10 -315
  9. package/esm2022/lib/aigc/avatar/comp-avatar-particle/index.mjs +10 -3
  10. package/esm2022/lib/aigc/avatar/comp-avatar-particle/role-points.class.mjs +10 -57
  11. package/esm2022/lib/aigc/avatar/comp-avatar-role-image/comp-avatar-role-image.component.mjs +10 -98
  12. package/esm2022/lib/aigc/avatar/comp-avatar-role-video/comp-avatar-role-video.component.mjs +10 -107
  13. package/esm2022/lib/aigc/avatar/comp-avatar-talk/comp-avatar-talk.component.mjs +10 -113
  14. package/esm2022/lib/aigc/avatar/index.mjs +10 -8
  15. package/esm2022/lib/aigc/avatar/interface-avatar-role.mjs +10 -2
  16. package/esm2022/lib/aigc/avatar/modal-chat-voice-input/modal-chat-voice-input.component.mjs +8 -164
  17. package/esm2022/lib/aigc/chat/chat-header-area/comp-header-area.component.mjs +10 -49
  18. package/esm2022/lib/aigc/chat/chat-header-area/index.mjs +10 -2
  19. package/esm2022/lib/aigc/chat/chat-list/chat-list.component.mjs +8 -155
  20. package/esm2022/lib/aigc/chat/chat-list/index.mjs +10 -2
  21. package/esm2022/lib/aigc/chat/chat-message-area/comp-message-area.component.mjs +10 -40
  22. package/esm2022/lib/aigc/chat/chat-message-area/index.mjs +10 -2
  23. package/esm2022/lib/aigc/chat/chat-message-card/comp-message-card.component.mjs +10 -140
  24. package/esm2022/lib/aigc/chat/chat-message-card/duration-str.pipe.mjs +10 -29
  25. package/esm2022/lib/aigc/chat/chat-message-card/index.mjs +10 -3
  26. package/esm2022/lib/aigc/chat/chat-modal-input/index.mjs +10 -3
  27. package/esm2022/lib/aigc/chat/chat-modal-input/modal-audio-message/modal-audio-message.component.mjs +8 -193
  28. package/esm2022/lib/aigc/chat/chat-modal-input/modal-input.component.mjs +10 -331
  29. package/esm2022/lib/aigc/chat/chat-panel/chat-panel.component.mjs +8 -249
  30. package/esm2022/lib/aigc/chat/comp-role-prompt/comp-role-prompt.component.mjs +10 -83
  31. package/esm2022/lib/aigc/chat/comp-role-prompt/index.mjs +10 -2
  32. package/esm2022/lib/aigc/chat/index.mjs +10 -8
  33. package/esm2022/lib/aigc/comp-markdown-preview/clipboard.service.mjs +10 -82
  34. package/esm2022/lib/aigc/comp-markdown-preview/markdown-parse.mjs +8 -367
  35. package/esm2022/lib/aigc/comp-markdown-preview/markdown-preview.component.mjs +10 -51
  36. package/esm2022/lib/aigc/comp-markdown-preview/markdown-preview.module.mjs +10 -24
  37. package/esm2022/lib/aigc/comp-markdown-preview/plugins/md-mathjax/index.mjs +10 -115
  38. package/esm2022/lib/aigc/index.mjs +10 -14
  39. package/esm2022/lib/aigc/service-fmai/fmai.service.mjs +10 -21
  40. package/esm2022/lib/aigc/service-fmai/service-chat/chat-class.mjs +10 -2
  41. package/esm2022/lib/aigc/service-fmai/service-chat/chat.service.mjs +8 -174
  42. package/esm2022/lib/aigc/service-fmai/service-chat/index.mjs +10 -7
  43. package/esm2022/lib/aigc/service-fmai/service-chat/mask-list.mjs +9 -194
  44. package/esm2022/lib/aigc/service-fmai/service-chat/pipes/chat-content.pipe.mjs +10 -27
  45. package/esm2022/lib/aigc/service-fmai/service-chat/pipes/hidexml.pipe.mjs +10 -27
  46. package/esm2022/lib/aigc/service-fmai/service-chat/utilnow.pipe.mjs +10 -68
  47. package/esm2022/lib/aigc/service-fmai/service-imagine/imagine-func.mjs +9 -162
  48. package/esm2022/lib/aigc/service-fmai/service-imagine/imagine-work.mjs +10 -68
  49. package/esm2022/lib/aigc/service-fmai/service-imagine/imagine.service.mjs +8 -313
  50. package/esm2022/lib/aigc/service-fmai/service-imagine/index.mjs +10 -4
  51. package/esm2022/lib/aigc/story/fm-office-viewer/fm-office-viewer.component.mjs +10 -62
  52. package/esm2022/lib/aigc/story/fm-story-card/fm-story-card.component.mjs +10 -87
  53. package/esm2022/lib/aigc/story/fm-story-list/fm-story-list.component.mjs +10 -345
  54. package/esm2022/lib/aigc/story/fm-story-list/story-preview.mjs +10 -81
  55. package/esm2022/lib/aigc/story/fm-story-loader/fm-story-loader.component.mjs +10 -152
  56. package/esm2022/lib/aigc/story/fm-story-splitter/fm-story-splitter.component.mjs +10 -42
  57. package/esm2022/lib/aigc/story/index.mjs +10 -6
  58. package/esm2022/lib/aigc/story/modal-chat-story/comp-chat-story-json/comp-chat-story-json.component.mjs +10 -47
  59. package/esm2022/lib/aigc/story/modal-chat-story/comp-diary-story/comp-diary-story.component.mjs +10 -95
  60. package/esm2022/lib/aigc/story/modal-chat-story/modal-chat-story.component.mjs +10 -253
  61. package/esm2022/lib/aigc/story/story.service.mjs +10 -35
  62. package/esm2022/lib/aigc/voice/fmode-voice.service.mjs +10 -636
  63. package/esm2022/lib/aigc/voice/index.mjs +10 -7
  64. package/esm2022/lib/aigc/voice/lib/audio/audio.player.mjs +10 -55
  65. package/esm2022/lib/aigc/voice/lib/audio/audio.streamer.mjs +10 -2
  66. package/esm2022/lib/aigc/voice/lib/audio/streamer.microsoft.mjs +10 -96
  67. package/esm2022/lib/aigc/voice/lib/audio/streamer.pcm.mjs +8 -177
  68. package/esm2022/lib/aigc/voice/lib/pcm2wav.mjs +10 -38
  69. package/esm2022/lib/aigc/voice/lib/recorder/extension-waveview.mjs +10 -215
  70. package/esm2022/lib/aigc/voice/lib/resample.mjs +10 -34
  71. package/esm2022/lib/aigc/voice/tts/fmode-tts-class.mjs +10 -189
  72. package/esm2022/lib/aigc/voice/tts/index.mjs +10 -5
  73. package/esm2022/lib/aigc/voice/tts/int-tts-provider.mjs +10 -2
  74. package/esm2022/lib/aigc/voice/tts/provider-doubao.mjs +8 -346
  75. package/esm2022/lib/code/fm-codemirror/fm-codemirror.component.mjs +8 -342
  76. package/esm2022/lib/code/index.mjs +10 -2
  77. package/esm2022/lib/core/agent/chat/completion/fmode-completion.mjs +8 -430
  78. package/esm2022/lib/core/agent/chat/completion/index.mjs +10 -2
  79. package/esm2022/lib/core/agent/chat/completion/int-gpt-chat-options.mjs +10 -2
  80. package/esm2022/lib/core/agent/chat/fmode-chat.mjs +8 -617
  81. package/esm2022/lib/core/agent/chat/index.mjs +10 -4
  82. package/esm2022/lib/core/agent/chat/interface.mjs +10 -2
  83. package/esm2022/lib/core/agent/index.mjs +10 -6
  84. package/esm2022/lib/core/agent/prompt/agent.prompt.mjs +10 -133
  85. package/esm2022/lib/core/agent/prompt/index.mjs +10 -2
  86. package/esm2022/lib/core/agent/prompt/prompt-util.mjs +10 -16
  87. package/esm2022/lib/core/agent/story/agent.story.mjs +10 -64
  88. package/esm2022/lib/core/agent/story/index.mjs +10 -2
  89. package/esm2022/lib/core/agent/task/agent.task.mjs +10 -90
  90. package/esm2022/lib/core/agent/task/index.mjs +10 -2
  91. package/esm2022/lib/core/agent/waiting/index.mjs +10 -3
  92. package/esm2022/lib/core/agent/waiting/loading/loading.ctrl.mjs +10 -279
  93. package/esm2022/lib/core/agent/waiting/tips/tips.ctrl.mjs +8 -190
  94. package/esm2022/lib/core/index.mjs +10 -4
  95. package/esm2022/lib/core/parse/datatype/acl.mjs +10 -57
  96. package/esm2022/lib/core/parse/datatype/file.mjs +10 -69
  97. package/esm2022/lib/core/parse/datatype/geopoint.mjs +10 -57
  98. package/esm2022/lib/core/parse/datatype/relation.mjs +10 -56
  99. package/esm2022/lib/core/parse/fmode.cloud.mjs +10 -0
  100. package/esm2022/lib/core/parse/fmode.object.mjs +10 -232
  101. package/esm2022/lib/core/parse/fmode.parse.mjs +10 -84
  102. package/esm2022/lib/core/parse/fmode.query.mjs +8 -500
  103. package/esm2022/lib/core/parse/fmode.user.mjs +10 -275
  104. package/esm2022/lib/core/parse/index.mjs +10 -5
  105. package/esm2022/lib/core/parse/types.mjs +10 -2
  106. package/esm2022/lib/core/voice/index.mjs +10 -3
  107. package/esm2022/lib/core/voice/tts/index.mjs +10 -2
  108. package/esm2022/lib/icon/filetype/audio.svg.mjs +10 -3
  109. package/esm2022/lib/icon/filetype/avatar.svg.mjs +10 -14
  110. package/esm2022/lib/icon/filetype/chat.svg.mjs +10 -3
  111. package/esm2022/lib/icon/filetype/docx.svg.mjs +10 -3
  112. package/esm2022/lib/icon/filetype/file.svg.mjs +10 -3
  113. package/esm2022/lib/icon/filetype/filetype.pipe.mjs +10 -57
  114. package/esm2022/lib/icon/filetype/index.mjs +10 -12
  115. package/esm2022/lib/icon/filetype/md.svg.mjs +10 -3
  116. package/esm2022/lib/icon/filetype/pdf.svg.mjs +10 -3
  117. package/esm2022/lib/icon/filetype/pptx.svg.mjs +10 -3
  118. package/esm2022/lib/icon/filetype/svgtoblob.mjs +10 -8
  119. package/esm2022/lib/icon/filetype/video.svg.mjs +10 -3
  120. package/esm2022/lib/icon/filetype/xlsx.svg.mjs +10 -3
  121. package/esm2022/lib/icon/index.mjs +10 -2
  122. package/esm2022/lib/map/comp-poi-picker/comp-poi-picker.component.mjs +10 -209
  123. package/esm2022/lib/map/comp-poi-picker/comp-poi-picker.module.mjs +10 -33
  124. package/esm2022/lib/map/index.mjs +10 -4
  125. package/esm2022/lib/map/map.module.mjs +10 -61
  126. package/esm2022/lib/map/page-loca-scatter/page-loca-scatter.component.mjs +10 -132
  127. package/esm2022/lib/map/page-map.start/page-map.start.component.mjs +8 -115
  128. package/esm2022/lib/map/page-plan-route/page-plan-route.component.mjs +8 -118
  129. package/esm2022/lib/nova-cloud/index.mjs +10 -3
  130. package/esm2022/lib/nova-cloud/ncloud-api-func.mjs +10 -91
  131. package/esm2022/lib/nova-cloud/nova-cloud.service.mjs +10 -53
  132. package/esm2022/lib/payment/index.mjs +10 -2
  133. package/esm2022/lib/payment/payment/payment.component.mjs +10 -543
  134. package/esm2022/lib/payment/payment.service.mjs +10 -202
  135. package/esm2022/lib/person/comp-person-gender-icon/comp-person-gender-icon.component.mjs +10 -48
  136. package/esm2022/lib/person/comp-person-item/comp-person-item.component.mjs +10 -29
  137. package/esm2022/lib/person/comp-person-story/comp-person-story.component.mjs +10 -211
  138. package/esm2022/lib/person/edit-upload/edit-upload.component.mjs +8 -412
  139. package/esm2022/lib/person/edit-upload/edit-upload.module.mjs +10 -50
  140. package/esm2022/lib/person/index.mjs +10 -5
  141. package/esm2022/lib/person/modal-person-select/modal-person-select.component.mjs +10 -144
  142. package/esm2022/lib/person/modal-user-verify/secret-text.pipe.mjs +10 -41
  143. package/esm2022/lib/person/modal-user-verify/user-verify.component.mjs +10 -595
  144. package/esm2022/lib/person/person-detail/person-detail.component.mjs +10 -172
  145. package/esm2022/lib/person/person.service.mjs +8 -193
  146. package/esm2022/lib/platform/cross.service.mjs +10 -62
  147. package/esm2022/lib/platform/index.mjs +10 -2
  148. package/esm2022/lib/social/index.mjs +10 -2
  149. package/esm2022/lib/social/wechat/wechat-jssdk.service.mjs +8 -230
  150. package/esm2022/lib/storage/comp-hwobs-manager/hwobs-manager.component.mjs +10 -59
  151. package/esm2022/lib/storage/index.mjs +10 -5
  152. package/esm2022/lib/storage/service-hwobs/hwobs.service.mjs +8 -131
  153. package/esm2022/lib/storage/service-hwobs/index.mjs +10 -3
  154. package/esm2022/lib/storage/service-hwobs/typings/esdk-obs-browser.mjs +1 -1
  155. package/esm2022/lib/storage/service-upload/index.mjs +10 -2
  156. package/esm2022/lib/storage/service-upload/nova-upload.service.mjs +8 -513
  157. package/esm2022/lib/storage/service-upload/util-file-md5.mjs +10 -28
  158. package/esm2022/lib/storage/service-upload/util-file-metadata.mjs +10 -93
  159. package/esm2022/lib/storage/storage.module.mjs +10 -42
  160. package/esm2022/lib/text/fm-article-editor/article-editor-topbar/article-editor-topbar.component.mjs +10 -72
  161. package/esm2022/lib/text/fm-article-editor/article.service.mjs +10 -237
  162. package/esm2022/lib/text/fm-article-editor/comp-upload-book-banners/comp-upload-book-banners.component.mjs +8 -72
  163. package/esm2022/lib/text/fm-article-editor/draft.service.mjs +10 -207
  164. package/esm2022/lib/text/fm-article-editor/fm-article-aitool/fm-article-aitool.component.mjs +8 -227
  165. package/esm2022/lib/text/fm-article-editor/fm-article-draft/fm-article-draft.component.mjs +10 -303
  166. package/esm2022/lib/text/fm-article-editor/fm-article-editor.component.mjs +8 -371
  167. package/esm2022/lib/text/fm-article-editor/fm-article-outline/fm-article-outline.component.mjs +10 -94
  168. package/esm2022/lib/text/fm-article-editor/fm-article-outline-leftitem/fm-article-outline-leftitem.component.mjs +10 -36
  169. package/esm2022/lib/text/fm-article-editor/fm-article-preview/fm-article-preview.component.mjs +10 -281
  170. package/esm2022/lib/text/fm-article-editor/fm-article-write-options/fm-article-write-options.component.mjs +10 -72
  171. package/esm2022/lib/text/fm-article-editor/outline-count.pipe.mjs +10 -22
  172. package/esm2022/lib/text/fm-article-editor/prompt/prompt-insertion-article.mjs +10 -169
  173. package/esm2022/lib/text/fm-article-editor/task-article-generation.mjs +10 -69
  174. package/esm2022/lib/text/fm-article-editor/tasks/task-article-draft-create.mjs +10 -65
  175. package/esm2022/lib/text/fm-article-editor/tasks/task-article-outline-edit.mjs +10 -55
  176. package/esm2022/lib/text/fm-article-editor/tasks/task-article-outline.mjs +10 -142
  177. package/esm2022/lib/text/fm-article-editor/tasks/task-article-preview.mjs +10 -35
  178. package/esm2022/lib/text/fm-article-editor/tasks/task-article-writing-options.mjs +10 -81
  179. package/esm2022/lib/text/fm-article-editor/tasks/task-document-select.mjs +10 -50
  180. package/esm2022/lib/text/fm-text-quill/fm-text-quill.component.mjs +10 -145
  181. package/esm2022/lib/text/index.mjs +10 -4
  182. package/esm2022/lib/user/account/account.service.mjs +10 -222
  183. package/esm2022/lib/user/captcha/captcha.component.mjs +10 -135
  184. package/esm2022/lib/user/comp-user-avatar/comp-user-avatar.component.mjs +10 -66
  185. package/esm2022/lib/user/index.mjs +10 -17
  186. package/esm2022/lib/user/login/auth.guard.mjs +10 -28
  187. package/esm2022/lib/user/login/auth.service.mjs +8 -433
  188. package/esm2022/lib/user/login/login.component.mjs +10 -916
  189. package/esm2022/lib/user/modal-user-login/modal-user-login.component.mjs +10 -311
  190. package/esm2022/lib/user/profile/auth-profile.guard.mjs +10 -27
  191. package/esm2022/lib/user/profile/auth-profile.service.mjs +10 -122
  192. package/esm2022/lib/user/profile/profile-bind/profile-bind.component.mjs +10 -164
  193. package/esm2022/lib/user/profile/profile-bind/profile-confirm-modal.component.mjs +10 -79
  194. package/esm2022/lib/user/profile/profile.module.mjs +10 -54
  195. package/esm2022/lib/user/staff/index.mjs +10 -4
  196. package/esm2022/lib/user/staff/staff.guard.mjs +10 -26
  197. package/esm2022/lib/user/staff/staff.module.mjs +10 -18
  198. package/esm2022/lib/user/staff/staff.service.mjs +10 -85
  199. package/esm2022/lib/user/user-name.pipe.mjs +10 -29
  200. package/esm2022/lib/user/user.module.mjs +10 -102
  201. package/esm2022/lib/video/fm-video/fm-video.component.mjs +10 -67
  202. package/esm2022/lib/video/index.mjs +10 -2
  203. package/esm2022/public-api.mjs +10 -21
  204. package/fesm2022/fmode-ng.mjs +7 -19788
  205. package/fesm2022/fmode-ng.mjs.map +1 -1
  206. package/lib/aigc/avatar/comp-avatar-role-image/comp-avatar-role-image.component.d.ts +2 -2
  207. package/lib/aigc/avatar/comp-avatar-role-video/comp-avatar-role-video.component.d.ts +2 -2
  208. package/lib/aigc/avatar/comp-avatar-talk/comp-avatar-talk.component.d.ts +2 -2
  209. package/lib/aigc/chat/chat-list/chat-list.component.d.ts +3 -2
  210. package/lib/aigc/chat/chat-message-card/comp-message-card.component.d.ts +3 -3
  211. package/lib/aigc/chat/chat-modal-input/modal-audio-message/modal-audio-message.component.d.ts +2 -2
  212. package/lib/aigc/chat/chat-modal-input/modal-input.component.d.ts +3 -3
  213. package/lib/aigc/service-fmai/service-imagine/imagine-func.d.ts +3 -3
  214. package/lib/aigc/service-fmai/service-imagine/imagine-work.d.ts +4 -4
  215. package/lib/aigc/service-fmai/service-imagine/imagine.service.d.ts +6 -6
  216. package/lib/aigc/story/fm-office-viewer/fm-office-viewer.component.d.ts +2 -2
  217. package/lib/aigc/story/fm-story-card/fm-story-card.component.d.ts +2 -1
  218. package/lib/aigc/story/fm-story-list/fm-story-list.component.d.ts +6 -6
  219. package/lib/aigc/story/fm-story-list/story-preview.d.ts +4 -4
  220. package/lib/aigc/story/fm-story-loader/fm-story-loader.component.d.ts +3 -3
  221. package/lib/aigc/story/fm-story-splitter/fm-story-splitter.component.d.ts +2 -2
  222. package/lib/aigc/story/modal-chat-story/comp-chat-story-json/comp-chat-story-json.component.d.ts +3 -2
  223. package/lib/aigc/story/modal-chat-story/comp-diary-story/comp-diary-story.component.d.ts +3 -2
  224. package/lib/aigc/story/modal-chat-story/modal-chat-story.component.d.ts +6 -6
  225. package/lib/aigc/story/story.service.d.ts +3 -2
  226. package/lib/aigc/voice/tts/fmode-tts-class.d.ts +2 -2
  227. package/lib/core/agent/chat/fmode-chat.d.ts +11 -11
  228. package/lib/core/agent/story/agent.story.d.ts +8 -8
  229. package/lib/core/parse/datatype/geopoint.d.ts +5 -1
  230. package/lib/core/parse/fmode.cloud.d.ts +32 -0
  231. package/lib/core/parse/fmode.object.d.ts +2 -2
  232. package/lib/core/parse/fmode.parse.d.ts +15 -9
  233. package/lib/core/parse/fmode.query.d.ts +41 -2
  234. package/lib/core/parse/fmode.user.d.ts +1 -0
  235. package/lib/core/parse/index.d.ts +1 -0
  236. package/lib/core/parse/types.d.ts +2 -0
  237. package/lib/map/comp-poi-picker/comp-poi-picker.component.d.ts +5 -5
  238. package/lib/payment/payment/payment.component.d.ts +3 -3
  239. package/lib/payment/payment.service.d.ts +4 -4
  240. package/lib/person/comp-person-gender-icon/comp-person-gender-icon.component.d.ts +2 -1
  241. package/lib/person/comp-person-item/comp-person-item.component.d.ts +2 -1
  242. package/lib/person/comp-person-story/comp-person-story.component.d.ts +13 -13
  243. package/lib/person/edit-upload/edit-upload.component.d.ts +1 -1
  244. package/lib/person/modal-person-select/modal-person-select.component.d.ts +5 -5
  245. package/lib/person/modal-user-verify/user-verify.component.d.ts +5 -5
  246. package/lib/person/person-detail/person-detail.component.d.ts +4 -4
  247. package/lib/person/person.service.d.ts +7 -7
  248. package/lib/storage/service-hwobs/hwobs.service.d.ts +4 -5
  249. package/lib/storage/service-upload/nova-upload.service.d.ts +5 -5
  250. package/lib/text/fm-article-editor/article-editor-topbar/article-editor-topbar.component.d.ts +2 -1
  251. package/lib/text/fm-article-editor/article.service.d.ts +12 -12
  252. package/lib/text/fm-article-editor/draft.service.d.ts +5 -5
  253. package/lib/text/fm-article-editor/fm-article-aitool/fm-article-aitool.component.d.ts +3 -2
  254. package/lib/text/fm-article-editor/fm-article-draft/fm-article-draft.component.d.ts +16 -17
  255. package/lib/text/fm-article-editor/fm-article-editor.component.d.ts +3 -3
  256. package/lib/text/fm-article-editor/fm-article-outline/fm-article-outline.component.d.ts +11 -10
  257. package/lib/text/fm-article-editor/fm-article-preview/fm-article-preview.component.d.ts +2 -2
  258. package/lib/text/fm-article-editor/prompt/prompt-insertion-article.d.ts +3 -3
  259. package/lib/text/fm-article-editor/task-article-generation.d.ts +5 -4
  260. package/lib/text/fm-article-editor/tasks/task-article-draft-create.d.ts +3 -3
  261. package/lib/text/fm-article-editor/tasks/task-article-outline-edit.d.ts +3 -3
  262. package/lib/text/fm-article-editor/tasks/task-article-outline.d.ts +3 -3
  263. package/lib/text/fm-article-editor/tasks/task-article-preview.d.ts +2 -1
  264. package/lib/text/fm-article-editor/tasks/task-article-writing-options.d.ts +3 -3
  265. package/lib/text/fm-article-editor/tasks/task-document-select.d.ts +3 -3
  266. package/lib/user/account/account.service.d.ts +2 -2
  267. package/lib/user/comp-user-avatar/comp-user-avatar.component.d.ts +2 -2
  268. package/lib/user/login/auth.service.d.ts +5 -5
  269. package/lib/user/login/login.component.d.ts +3 -3
  270. package/lib/user/profile/auth-profile.service.d.ts +8 -8
  271. package/lib/user/profile/profile-bind/profile-bind.component.d.ts +7 -7
  272. package/lib/user/profile/profile-bind/profile-confirm-modal.component.d.ts +2 -2
  273. package/lib/user/staff/staff.service.d.ts +3 -3
  274. package/lib/user/user-name.pipe.d.ts +2 -2
  275. package/package.json +12 -18
@@ -1,636 +1,10 @@
1
- // import RecorderManager from "./lib/xunfei-recorder"
2
- // import {RecorderManager} from "./lib/recorder/recorder-manager"
3
- import Recorder from 'recorder-core';
4
- // 检测浏览器支持情况
5
- function checkWorkletSupport() {
6
- try {
7
- return (window.AudioWorkletNode &&
8
- window.AudioContext &&
9
- window.AudioContext.audioWorklet &&
10
- !/Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent));
11
- }
12
- catch (e) {
13
- return false;
14
- }
15
- }
16
- Recorder.ConnectEnableWorklet = checkWorkletSupport(); // 明确禁用AudioWorklet 兼容不支持的场景
17
- import 'recorder-core/src/engine/pcm';
18
- import 'recorder-core/src/engine/wav';
19
- import 'recorder-core/src/extensions/waveview';
20
- // import './lib/recorder/engine-pcm'
21
- // import './lib/recorder/engine-wav'
22
- // import './lib/recorder/extension-waveview'
23
- import CryptoJS from "crypto-js";
24
- import { pcmtoWav } from './lib/pcm2wav';
25
- import { convertFrameBufferToBase64, resampleBuffer } from './lib/resample';
26
- // import { FmodeTTSXunfei } fro./class-tts-xunfei.ts.bakfei'
27
- export class FmodeVoiceService {
28
- constructor(platform, diagnostic) {
29
- this.platform = platform;
30
- this.diagnostic = diagnostic;
31
- this.disableASR = false;
32
- // webSpeech = WebSpeech;
33
- // 状态管理
34
- this.isRecording = false;
35
- this.isUserFinish = false;
36
- this.shouldReconnect = true;
37
- this.reconnectAttempts = 0;
38
- this.maxReconnectAttempts = 3;
39
- this.reconnectDelay = 1000;
40
- // 录制相关 - 独立于WebSocket
41
- this.recordWavBlob = null;
42
- this.recordPcmBlob = null;
43
- this.recordDuration = 0;
44
- this.allRecordedBuffers = []; // 存储所有录制的buffer
45
- this.currentRecorder = null; // 当前的录制器实例
46
- this.btnStatus = "UNDEFINED";
47
- this.connStatus = "";
48
- // 结果相关
49
- this.resultText = "";
50
- this.resultTextTemp = "";
51
- this.durationStr = "00:00";
52
- this.duration = 0;
53
- this.recordType = "pcm";
54
- this.encodingType = "raw";
55
- // API配置
56
- this.APPID = "50f4a46c";
57
- this.API_SECRET = "NzFlNmFhZDJjMDNkZGM3NzI0Mzg2OGNm";
58
- this.API_KEY = "106ddc40dfd4b9ca6d7b47c70fada749";
59
- this.requestPermission();
60
- }
61
- // 用户操作方法
62
- toggleRecord() {
63
- console.log('toggleRecord', this.btnStatus);
64
- if (this.btnStatus === "UNDEFINED" || this.btnStatus === "CLOSED") {
65
- this.startTalk();
66
- }
67
- else if (this.btnStatus === "CONNECTING" || this.btnStatus === "OPEN") {
68
- this.finishTalk();
69
- }
70
- }
71
- async startTalk(event) {
72
- console.log('startTalk called');
73
- this.resetState();
74
- this.onBeforeStartTalk && this.onBeforeStartTalk();
75
- event?.preventDefault();
76
- try {
77
- await this.openWithPriviledge();
78
- // 1. 首先启动独立录制
79
- await this.startIndependentRecording();
80
- // 2. 然后启动WebSocket连接
81
- this.isRecording = true;
82
- this.shouldReconnect = true;
83
- this.reconnectAttempts = 0;
84
- setTimeout(() => {
85
- this.connectWebSocket();
86
- }, 100);
87
- this.startCountdown();
88
- this.onAfterStartTalk && this.onAfterStartTalk();
89
- }
90
- catch (error) {
91
- console.error('Failed to start talk:', error);
92
- this.resetState();
93
- }
94
- }
95
- async finishTalk() {
96
- console.log('finishTalk called');
97
- this.isUserFinish = true;
98
- this.shouldReconnect = false;
99
- this.onBeforeFinishTalk && this.onBeforeFinishTalk();
100
- try {
101
- // 1. 先关闭WebSocket
102
- this.closeWebSocket();
103
- // 2. 停止独立录制并获取完整音频
104
- await this.stopIndependentRecording();
105
- // 3. 清理状态
106
- this.isRecording = false;
107
- this.clearTimers();
108
- this.changeBtnStatus("CLOSED");
109
- // 4. 延迟回调确保所有处理完成
110
- setTimeout(() => {
111
- if (this.isUserFinish) {
112
- this.onAfterFinishTalk && this.onAfterFinishTalk();
113
- this.isUserFinish = false;
114
- }
115
- }, 500);
116
- }
117
- catch (error) {
118
- console.error('Error finishing talk:', error);
119
- }
120
- }
121
- cancelTalk() {
122
- console.log('cancelTalk called');
123
- this.onBeforeCancelTalk && this.onBeforeCancelTalk();
124
- this.isUserFinish = false;
125
- this.shouldReconnect = false;
126
- this.isRecording = false;
127
- // 关闭WebSocket
128
- this.closeWebSocket();
129
- // 停止独立录制但不保存结果
130
- this.cancelIndependentRecording();
131
- this.resetState();
132
- this.onAfterCancelTalk && this.onAfterCancelTalk();
133
- }
134
- // 独立录制方法 - 与WebSocket完全解耦
135
- async startIndependentRecording() {
136
- console.log('Starting independent recording');
137
- this.allRecordedBuffers = [];
138
- this.createIndependentRecorder();
139
- return new Promise((resolve, reject) => {
140
- this.currentRecorder.open(() => {
141
- console.log('Independent recorder opened');
142
- // 创建可视化
143
- let waveDiv = document.querySelector(".record-wave");
144
- if (waveDiv && Recorder.WaveView) {
145
- this.waveClient = Recorder.WaveView({ elem: ".record-wave" });
146
- }
147
- this.currentRecorder.start();
148
- this.onAfterRecordStart && this.onAfterRecordStart();
149
- resolve();
150
- }, (msg, isUserNotAllow) => {
151
- console.error('Failed to open independent recorder:', msg);
152
- reject(new Error(msg));
153
- });
154
- });
155
- }
156
- async stopIndependentRecording() {
157
- console.log('Stopping independent recording');
158
- if (!this.currentRecorder) {
159
- console.log('No independent recorder to stop');
160
- return;
161
- }
162
- return new Promise((resolve) => {
163
- this.currentRecorder.stop(async (blob, duration) => {
164
- console.log('Independent recording stopped successfully', blob, duration);
165
- // 保存完整的录制结果
166
- this.recordPcmBlob = blob;
167
- this.recordWavBlob = await this.pcmBlobToWavBlob(blob, 44100);
168
- // 清理录制器
169
- this.cleanupIndependentRecorder();
170
- resolve();
171
- }, (msg) => {
172
- console.error("独立录音停止失败:" + msg);
173
- this.cleanupIndependentRecorder();
174
- resolve();
175
- });
176
- });
177
- }
178
- cancelIndependentRecording() {
179
- console.log('Cancelling independent recording');
180
- if (this.currentRecorder) {
181
- try {
182
- // 直接关闭,不保存结果
183
- this.currentRecorder.close();
184
- }
185
- catch (error) {
186
- console.error('Error closing independent recorder:', error);
187
- }
188
- }
189
- this.cleanupIndependentRecorder();
190
- this.recordPcmBlob = null;
191
- this.recordWavBlob = null;
192
- }
193
- createIndependentRecorder() {
194
- if (this.currentRecorder) {
195
- this.cleanupIndependentRecorder();
196
- }
197
- this.currentRecorder = Recorder({
198
- type: this.recordType,
199
- sampleRate: 44100,
200
- bitRate: 16,
201
- onProcess: (buffers, powerLevel, bufferDuration, bufferSampleRate, newBufferIdx, asyncEnd) => {
202
- if (!this.isRecording)
203
- return;
204
- // 保存所有buffer用于最终合成完整音频
205
- this.allRecordedBuffers = buffers;
206
- // 获取最新的音频片段用于实时ASR
207
- let frameBuffer = buffers.length && buffers[buffers.length - 1];
208
- if (!frameBuffer)
209
- return;
210
- // 重采样到16kHz用于ASR
211
- frameBuffer = resampleBuffer(frameBuffer, 44100, 16000);
212
- // 发送到WebSocket进行实时ASR
213
- this.sendAudioToWebSocket(frameBuffer);
214
- // 更新波形显示
215
- this.waveClient?.input(buffers[buffers.length - 1], powerLevel, bufferSampleRate);
216
- }
217
- });
218
- }
219
- cleanupIndependentRecorder() {
220
- if (this.currentRecorder) {
221
- try {
222
- this.currentRecorder.close();
223
- }
224
- catch (error) {
225
- console.error('Error cleaning up independent recorder:', error);
226
- }
227
- this.currentRecorder = null;
228
- }
229
- }
230
- sendAudioToWebSocket(frameBuffer) {
231
- if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN && !this.disableASR) {
232
- try {
233
- this.iatWS.send(JSON.stringify({
234
- data: {
235
- status: 1,
236
- format: "audio/L16;rate=16000",
237
- encoding: this.encodingType,
238
- audio: convertFrameBufferToBase64(frameBuffer)
239
- },
240
- }));
241
- }
242
- catch (error) {
243
- console.error("Error sending audio data to WebSocket:", error);
244
- }
245
- }
246
- }
247
- // WebSocket相关方法 - 仅用于ASR,不控制录制
248
- connectWebSocket() {
249
- console.log("connectWebSocket called, shouldReconnect:", this.shouldReconnect);
250
- if (!this.shouldReconnect || !this.isRecording) {
251
- console.log("Not connecting WebSocket - shouldReconnect:", this.shouldReconnect, "isRecording:", this.isRecording);
252
- return;
253
- }
254
- // 关闭现有连接
255
- this.closeWebSocket();
256
- const websocketUrl = this.getWebSocketUrl();
257
- console.log("Connecting to:", websocketUrl);
258
- try {
259
- if ("WebSocket" in window) {
260
- this.iatWS = new WebSocket(websocketUrl);
261
- }
262
- else {
263
- console.error("浏览器不支持WebSocket");
264
- alert("浏览器不支持WebSocket");
265
- return;
266
- }
267
- this.changeBtnStatus("CONNECTING");
268
- this.iatWS.onopen = (e) => {
269
- console.log("WebSocket connected for ASR");
270
- this.reconnectAttempts = 0;
271
- this.changeBtnStatus("OPEN");
272
- // 发送初始参数
273
- const params = {
274
- common: {
275
- app_id: this.APPID,
276
- },
277
- business: {
278
- language: "zh_cn",
279
- domain: "iat",
280
- accent: "mandarin",
281
- vad_eos: 10000, // 增加到10秒
282
- dwa: "wpgs",
283
- },
284
- data: {
285
- status: 0,
286
- format: "audio/L16;rate=16000",
287
- encoding: this.encodingType,
288
- },
289
- };
290
- this.iatWS.send(JSON.stringify(params));
291
- };
292
- this.iatWS.onmessage = (e) => {
293
- this.renderResult(e.data);
294
- };
295
- this.iatWS.onerror = (e) => {
296
- console.error("WebSocket error:", e);
297
- this.handleWebSocketError();
298
- };
299
- this.iatWS.onclose = (e) => {
300
- console.log("WebSocket closed:", e.code, e.reason);
301
- this.handleWebSocketClose(e);
302
- };
303
- }
304
- catch (error) {
305
- console.error("Failed to create WebSocket:", error);
306
- this.handleWebSocketError();
307
- }
308
- }
309
- handleWebSocketClose(event) {
310
- console.log("handleWebSocketClose", {
311
- code: event.code,
312
- reason: event.reason,
313
- shouldReconnect: this.shouldReconnect,
314
- isRecording: this.isRecording,
315
- isUserFinish: this.isUserFinish
316
- });
317
- // 如果用户已经结束或不应该重连,则不处理
318
- if (this.isUserFinish || !this.shouldReconnect || !this.isRecording) {
319
- return;
320
- }
321
- // 发送结束帧(如果需要)
322
- try {
323
- if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {
324
- this.iatWS.send(JSON.stringify({
325
- "data": {
326
- "status": 2,
327
- "format": "audio/L16;rate=16000",
328
- "encoding": this.encodingType,
329
- "audio": ""
330
- }
331
- }));
332
- }
333
- }
334
- catch (err) {
335
- console.error('Error sending end frame on close:', err);
336
- }
337
- // 正常关闭码或服务端主动关闭,尝试重连
338
- if (event.code === 1000 || event.code === 1006 || event.code === 1011) {
339
- this.attemptReconnect();
340
- }
341
- else {
342
- console.error("WebSocket closed with error code:", event.code);
343
- this.handleWebSocketError();
344
- }
345
- }
346
- handleWebSocketError() {
347
- if (!this.shouldReconnect || !this.isRecording) {
348
- return;
349
- }
350
- this.attemptReconnect();
351
- }
352
- attemptReconnect() {
353
- if (this.reconnectAttempts >= this.maxReconnectAttempts) {
354
- console.error("Max reconnect attempts reached");
355
- // 不改变录制状态,只是ASR暂时不可用
356
- return;
357
- }
358
- this.reconnectAttempts++;
359
- console.log(`Attempting WebSocket reconnect ${this.reconnectAttempts}/${this.maxReconnectAttempts}`);
360
- setTimeout(() => {
361
- if (this.shouldReconnect && this.isRecording && !this.isUserFinish) {
362
- this.connectWebSocket();
363
- }
364
- }, this.reconnectDelay * this.reconnectAttempts);
365
- }
366
- closeWebSocket() {
367
- if (this.iatWS) {
368
- try {
369
- // 发送结束帧
370
- if (this.iatWS.readyState === WebSocket.OPEN) {
371
- this.iatWS.send(JSON.stringify({
372
- "data": {
373
- "status": 2,
374
- "format": "audio/L16;rate=16000",
375
- "encoding": this.encodingType,
376
- "audio": ""
377
- }
378
- }));
379
- }
380
- if (this.iatWS.readyState === WebSocket.OPEN || this.iatWS.readyState === WebSocket.CONNECTING) {
381
- this.iatWS.close(1000, "User initiated close");
382
- }
383
- }
384
- catch (error) {
385
- console.error("Error closing WebSocket:", error);
386
- }
387
- this.iatWS = null;
388
- }
389
- }
390
- // 状态管理
391
- resetState() {
392
- this.resultText = "";
393
- this.resultTextTemp = "";
394
- this.durationStr = "00:00";
395
- this.duration = 0;
396
- this.recordDuration = 0;
397
- this.allRecordedBuffers = [];
398
- this.clearTimers();
399
- }
400
- clearTimers() {
401
- if (this.countdownInterval) {
402
- clearInterval(this.countdownInterval);
403
- this.countdownInterval = null;
404
- }
405
- if (this.durationInterval) {
406
- clearInterval(this.durationInterval);
407
- this.durationInterval = null;
408
- }
409
- }
410
- changeBtnStatus(status) {
411
- this.btnStatus = status;
412
- switch (status) {
413
- case "CONNECTING":
414
- this.connStatus = "建立连接中";
415
- break;
416
- case "OPEN":
417
- this.connStatus = `录音中(${this.durationStr})`;
418
- break;
419
- case "CLOSING":
420
- this.connStatus = "关闭连接中";
421
- break;
422
- case "CLOSED":
423
- this.connStatus = "开始录音";
424
- break;
425
- }
426
- }
427
- // 计时器
428
- startCountdown() {
429
- this.clearTimers();
430
- this.recordDuration = 0;
431
- this.duration = 0;
432
- this.now = new Date();
433
- // 录音时长计时器(100ms精度)
434
- this.durationInterval = setInterval(() => {
435
- if (this.isRecording) {
436
- this.recordDuration += 100;
437
- }
438
- }, 100);
439
- // 显示时长计时器(1秒精度)
440
- this.countdownInterval = setInterval(() => {
441
- if (this.isRecording) {
442
- this.countTimer();
443
- }
444
- }, 1000);
445
- }
446
- countTimer() {
447
- this.duration++;
448
- let minuteStr = String(Math.floor(this.duration / 60)).padStart(2, "0");
449
- let secondStr = String(this.duration % 60).padStart(2, "0");
450
- let durationStr = minuteStr + ":" + secondStr;
451
- this.durationStr = durationStr;
452
- this.connStatus = `录音中(${this.durationStr})`;
453
- this.onDurationStrChange && this.onDurationStrChange(durationStr);
454
- }
455
- // 结果处理
456
- renderResult(resultData) {
457
- try {
458
- let jsonData = JSON.parse(resultData);
459
- if (jsonData.data && jsonData.data.result) {
460
- let data = jsonData.data.result;
461
- let str = "";
462
- let ws = data.ws;
463
- for (let i = 0; i < ws.length; i++) {
464
- str = str + ws[i].cw[0].w;
465
- }
466
- // 处理动态修正
467
- if (data.pgs) {
468
- if (data.pgs === "apd") {
469
- this.resultText = this.resultTextTemp;
470
- }
471
- this.resultTextTemp = this.resultText + str;
472
- }
473
- else {
474
- this.resultText = this.resultText + str;
475
- }
476
- this.onInputChange && this.onInputChange(this.getUserInput());
477
- }
478
- // 处理结束状态
479
- if (jsonData.code === 0 && jsonData.data.status === 2) {
480
- console.log("ASR session completed, will reconnect if still recording");
481
- if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {
482
- this.iatWS.close();
483
- }
484
- }
485
- // 处理错误
486
- if (jsonData.code !== 0) {
487
- console.error("ASR error:", jsonData);
488
- if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {
489
- this.iatWS.close();
490
- }
491
- }
492
- }
493
- catch (error) {
494
- console.error("Error parsing result:", error);
495
- }
496
- }
497
- getUserInput() {
498
- return "" + (this.resultTextTemp || this.resultText);
499
- }
500
- // 权限和初始化
501
- async openWithPriviledge() {
502
- await this.requestPermission();
503
- if (Recorder.IsOpen())
504
- return true;
505
- // 这里不创建录制器,因为我们使用独立的录制器
506
- return true;
507
- }
508
- // 音频处理方法
509
- async pcmBlobToWavBlob(pcmBlob, sampleRate) {
510
- return new Promise(resolve => {
511
- let fileReader = new FileReader();
512
- fileReader.onload = function (event) {
513
- let pcmData = event.target.result;
514
- let wavBlob = pcmtoWav(pcmData, sampleRate, 1, 16);
515
- resolve(wavBlob);
516
- };
517
- fileReader.readAsArrayBuffer(pcmBlob);
518
- });
519
- }
520
- async playPCM(pcmBlob, sampleRate) {
521
- let wavBlob = await this.pcmBlobToWavBlob(pcmBlob, sampleRate);
522
- let wavUrl = window.URL.createObjectURL(wavBlob);
523
- let audio = new Audio();
524
- audio.src = wavUrl;
525
- audio.play();
526
- }
527
- playRecord() {
528
- if (this.recordPcmBlob) {
529
- this.playPCM(this.recordPcmBlob, 44100);
530
- }
531
- }
532
- // WebSocket URL生成
533
- getWebSocketUrl() {
534
- let url = "wss://iat-api.xfyun.cn/v2/iat";
535
- let host = "iat-api.xfyun.cn";
536
- let apiKey = this.API_KEY;
537
- let apiSecret = this.API_SECRET;
538
- let date = new Date().toUTCString();
539
- let algorithm = "hmac-sha256";
540
- let headers = "host date request-line";
541
- let signatureOrigin = `host: ${host}\ndate: ${date}\nGET /v2/iat HTTP/1.1`;
542
- let signatureSha = CryptoJS.HmacSHA256(signatureOrigin, apiSecret);
543
- let signature = CryptoJS.enc.Base64.stringify(signatureSha);
544
- let authorizationOrigin = `api_key="${apiKey}", algorithm="${algorithm}", headers="${headers}", signature="${signature}"`;
545
- let authorization = btoa(authorizationOrigin);
546
- url = `${url}?authorization=${authorization}&date=${date}&host=${host}`;
547
- return url;
548
- }
549
- toBase64(buffer) {
550
- var binary = "";
551
- var bytes = new Uint8Array(buffer);
552
- var len = bytes.byteLength;
553
- for (var i = 0; i < len; i++) {
554
- binary += String.fromCharCode(bytes[i]);
555
- }
556
- return window.btoa(binary);
557
- }
558
- // 移动端权限方法
559
- isCapacitor() {
560
- if (!this.platform?.is)
561
- return false;
562
- return this.platform.is("capacitor") || this.platform.is("cordova");
563
- }
564
- async requestPermission() {
565
- if (!this.isCapacitor())
566
- return;
567
- try {
568
- await this.requestStoagePermission();
569
- await this.requestCameraPermission();
570
- await this.requestMicPermission();
571
- await this.requestRecordAudioPermission();
572
- }
573
- catch (err) {
574
- console.error(err);
575
- }
576
- }
577
- async requestRecordAudioPermission() {
578
- let data = await this.diagnostic.requestRuntimePermissions([this.diagnostic.permission.RECORD_AUDIO]);
579
- console.log("record permission request:", data);
580
- return;
581
- }
582
- async requestMicPermission() {
583
- let isAvailable = await this.diagnostic.isMicrophoneAuthorized();
584
- console.log("permisson_MIC:", isAvailable);
585
- if (!isAvailable) {
586
- let data = await this.diagnostic.requestMicrophoneAuthorization();
587
- }
588
- return;
589
- }
590
- async requestStoagePermission() {
591
- let isAvailable = await this.diagnostic.isExternalStorageAuthorized();
592
- console.log("permisson_STORAGE:", isAvailable);
593
- if (!isAvailable) {
594
- let data = await this.diagnostic.requestExternalStorageAuthorization();
595
- }
596
- return;
597
- }
598
- async requestCameraPermission() {
599
- let isAvailable = await this.diagnostic.isCameraAuthorized();
600
- console.log("permisson_Camera:", isAvailable);
601
- if (!isAvailable) {
602
- let data = await this.diagnostic.requestCameraAuthorization();
603
- }
604
- return;
605
- }
606
- // 其他辅助方法
607
- splitAudioData(audioData) {
608
- const segmentSize = 1280;
609
- const segmentCount = Math.ceil(audioData.length / segmentSize);
610
- const segments = [];
611
- for (let i = 0; i < segmentCount; i++) {
612
- const start = i * segmentSize;
613
- const end = start + segmentSize;
614
- const segment = audioData.slice(start, end);
615
- segments.push(segment);
616
- }
617
- return segments;
618
- }
619
- BufferToBlob(buffer) {
620
- return new Blob([buffer], { type: 'audio/pcm' });
621
- }
622
- async playBuffers() {
623
- let audioBlob = await this.BuffersToBlob(this.allRecordedBuffers);
624
- this.playPCM(audioBlob, 44100);
625
- }
626
- BuffersToBlob(buffers) {
627
- let audioBuffer = [];
628
- buffers.forEach(buffer => {
629
- buffer.forEach(int16 => {
630
- audioBuffer.push(int16);
631
- });
632
- });
633
- return new Blob([audioBuffer], { type: 'audio/pcm' });
634
- }
635
- }
636
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm1vZGUtdm9pY2Uuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Ztb2RlLW5nL3NyYy9saWIvYWlnYy92b2ljZS9mbW9kZS12b2ljZS5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLHNEQUFzRDtBQUN0RCxrRUFBa0U7QUFDbEUsT0FBTyxRQUFRLE1BQU0sZUFBZSxDQUFBO0FBRXBDLFlBQVk7QUFDWixTQUFTLG1CQUFtQjtJQUMxQixJQUFJLENBQUM7UUFDSCxPQUFPLENBQ0wsTUFBTSxDQUFDLGdCQUFnQjtZQUN2QixNQUFNLENBQUMsWUFBWTtZQUNuQixNQUFNLENBQUMsWUFBWSxDQUFDLFlBQVk7WUFDaEMsQ0FBQyxnRUFBZ0UsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxDQUM1RixDQUFDO0lBQ0osQ0FBQztJQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7UUFDWCxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7QUFDSCxDQUFDO0FBQ0QsUUFBUSxDQUFDLG9CQUFvQixHQUFHLG1CQUFtQixFQUFFLENBQUMsQ0FBQyw0QkFBNEI7QUFFbkYsT0FBTyw4QkFBOEIsQ0FBQTtBQUNyQyxPQUFPLDhCQUE4QixDQUFBO0FBQ3JDLE9BQU8sdUNBQXVDLENBQUE7QUFDOUMscUNBQXFDO0FBQ3JDLHFDQUFxQztBQUNyQyw2Q0FBNkM7QUFDN0MsT0FBTyxRQUFRLE1BQU0sV0FBVyxDQUFBO0FBQ2hDLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLDBCQUEwQixFQUFpQixjQUFjLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUszRiw2REFBNkQ7QUFFN0QsTUFBTSxPQUFPLGlCQUFpQjtJQXlENUIsWUFDVSxRQUFrQixFQUNsQixVQUFzQjtRQUR0QixhQUFRLEdBQVIsUUFBUSxDQUFVO1FBQ2xCLGVBQVUsR0FBVixV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QUUsQ0FBQyxDQUFDO0lBQ3hELENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8vIGltcG9ydCBSZWNvcmRlck1hbmFnZXIgZnJvbSBcIi4vbGliL3h1bmZlaS1yZWNvcmRlclwiXHJcbi8vIGltcG9ydCB7UmVjb3JkZXJNYW5hZ2VyfSBmcm9tIFwiLi9saWIvcmVjb3JkZXIvcmVjb3JkZXItbWFuYWdlclwiXHJcbmltcG9ydCBSZWNvcmRlciBmcm9tICdyZWNvcmRlci1jb3JlJ1xyXG5cclxuLy8g5qOA5rWL5rWP6KeI5Zmo5pSv5oyB5oOF5Ya1XHJcbmZ1bmN0aW9uIGNoZWNrV29ya2xldFN1cHBvcnQoKSB7XHJcbiAgdHJ5IHtcclxuICAgIHJldHVybiAoXHJcbiAgICAgIHdpbmRvdy5BdWRpb1dvcmtsZXROb2RlICYmXHJcbiAgICAgIHdpbmRvdy5BdWRpb0NvbnRleHQgJiYgXHJcbiAgICAgIHdpbmRvdy5BdWRpb0NvbnRleHQuYXVkaW9Xb3JrbGV0ICYmXHJcbiAgICAgICEvQW5kcm9pZHx3ZWJPU3xpUGhvbmV8aVBhZHxpUG9kfEJsYWNrQmVycnl8SUVNb2JpbGV8T3BlcmEgTWluaS9pLnRlc3QobmF2aWdhdG9yLnVzZXJBZ2VudClcclxuICAgICk7XHJcbiAgfSBjYXRjaCAoZSkge1xyXG4gICAgcmV0dXJuIGZhbHNlO1xyXG4gIH1cclxufVxyXG5SZWNvcmRlci5Db25uZWN0RW5hYmxlV29ya2xldCA9IGNoZWNrV29ya2xldFN1cHBvcnQoKTsgLy8g5piO56Gu56aB55SoQXVkaW9Xb3JrbGV0IOWFvOWuueS4jeaUr+aMgeeahOWcuuaZr1xyXG5cclxuaW1wb3J0ICdyZWNvcmRlci1jb3JlL3NyYy9lbmdpbmUvcGNtJ1xyXG5pbXBvcnQgJ3JlY29yZGVyLWNvcmUvc3JjL2VuZ2luZS93YXYnXHJcbmltcG9ydCAncmVjb3JkZXItY29yZS9zcmMvZXh0ZW5zaW9ucy93YXZldmlldydcclxuLy8gaW1wb3J0ICcuL2xpYi9yZWNvcmRlci9lbmdpbmUtcGNtJ1xyXG4vLyBpbXBvcnQgJy4vbGliL3JlY29yZGVyL2VuZ2luZS13YXYnXHJcbi8vIGltcG9ydCAnLi9saWIvcmVjb3JkZXIvZXh0ZW5zaW9uLXdhdmV2aWV3J1xyXG5pbXBvcnQgQ3J5cHRvSlMgZnJvbSBcImNyeXB0by1qc1wiXHJcbmltcG9ydCB7IHBjbXRvV2F2IH0gZnJvbSAnLi9saWIvcGNtMndhdic7XHJcbmltcG9ydCB7IGNvbnZlcnRGcmFtZUJ1ZmZlclRvQmFzZTY0LCByZXNhbXBsZUF1ZGlvLCByZXNhbXBsZUJ1ZmZlciB9IGZyb20gJy4vbGliL3Jlc2FtcGxlJztcclxuLy8gaW1wb3J0IHsgV2ViU3BlZWNoIH0gZnJvbSAnLi9jbGFzcy1hc3InO1xyXG5pbXBvcnQgeyBQbGF0Zm9ybSB9IGZyb20gJ0Bpb25pYy9hbmd1bGFyJztcclxuLy8gaW1wb3J0IHsgRGlhZ25vc3RpYyB9IGZyb20gJ0Bhd2Vzb21lLWNvcmRvdmEtcGx1Z2lucy9kaWFnbm9zdGljL25neCc7XHJcbmltcG9ydCB7IERpYWdub3N0aWMgfSBmcm9tICdAYXdlc29tZS1jb3Jkb3ZhLXBsdWdpbnMvZGlhZ25vc3RpYy9uZ3gnO1xyXG4vLyBpbXBvcnQgeyBGbW9kZVRUU1h1bmZlaSB9IGZyby4vY2xhc3MtdHRzLXh1bmZlaS50cy5iYWtmZWknXHJcblxyXG5leHBvcnQgY2xhc3MgRm1vZGVWb2ljZVNlcnZpY2Uge1xyXG4gIGRpc2FibGVBU1I6IGJvb2xlYW4gPSBmYWxzZTtcclxuICAvLyB3ZWJTcGVlY2ggPSBXZWJTcGVlY2g7XHJcblxyXG4gIC8vIOeKtuaAgeeuoeeQhlxyXG4gIHByaXZhdGUgaXNSZWNvcmRpbmc6IGJvb2xlYW4gPSBmYWxzZTtcclxuICBwcml2YXRlIGlzVXNlckZpbmlzaDogYm9vbGVhbiA9IGZhbHNlO1xyXG4gIHByaXZhdGUgc2hvdWxkUmVjb25uZWN0OiBib29sZWFuID0gdHJ1ZTtcclxuICBwcml2YXRlIHJlY29ubmVjdEF0dGVtcHRzOiBudW1iZXIgPSAwO1xyXG4gIHByaXZhdGUgbWF4UmVjb25uZWN0QXR0ZW1wdHM6IG51bWJlciA9IDM7XHJcbiAgcHJpdmF0ZSByZWNvbm5lY3REZWxheTogbnVtYmVyID0gMTAwMDtcclxuXHJcbiAgLy8g5b2V5Yi255u45YWzIC0g54us56uL5LqOV2ViU29ja2V0XHJcbiAgcmVjb3JkV2F2QmxvYjogQmxvYiA9IG51bGw7XHJcbiAgcmVjb3JkUGNtQmxvYjogQmxvYiA9IG51bGw7XHJcbiAgcmVjb3JkRHVyYXRpb246IG51bWJlciA9IDA7XHJcbiAgcHJpdmF0ZSBhbGxSZWNvcmRlZEJ1ZmZlcnM6IEFycmF5PGFueT4gPSBbXTsgLy8g5a2Y5YKo5omA5pyJ5b2V5Yi255qEYnVmZmVyXHJcbiAgcHJpdmF0ZSBjdXJyZW50UmVjb3JkZXI6IGFueSA9IG51bGw7IC8vIOW9k+WJjeeahOW9leWItuWZqOWunuS+i1xyXG4gIFxyXG4gIC8vIFdlYlNvY2tldOebuOWFs1xyXG4gIGlhdFdTOiBXZWJTb2NrZXQ7XHJcbiAgYnRuU3RhdHVzID0gXCJVTkRFRklORURcIjtcclxuICBjb25uU3RhdHVzID0gXCJcIjtcclxuICBcclxuICAvLyDnu5Pmnpznm7jlhbNcclxuICByZXN1bHRUZXh0ID0gXCJcIjtcclxuICByZXN1bHRUZXh0VGVtcCA9IFwiXCI7XHJcbiAgXHJcbiAgLy8g6K6h5pe255u45YWzXHJcbiAgY291bnRkb3duSW50ZXJ2YWw6IGFueTtcclxuICBkdXJhdGlvbkludGVydmFsOiBhbnk7XHJcbiAgZHVyYXRpb25TdHI6IHN0cmluZyA9IFwiMDA6MDBcIjtcclxuICBkdXJhdGlvbjogbnVtYmVyID0gMDtcclxuICBub3c6IERhdGU7XHJcbiAgXHJcbiAgLy8g5rOi5b2i55u45YWzXHJcbiAgd2F2ZUNsaWVudDogYW55O1xyXG4gIHJlY29yZFR5cGUgPSBcInBjbVwiO1xyXG4gIGVuY29kaW5nVHlwZSA9IFwicmF3XCI7XHJcblxyXG4gIC8vIOWbnuiwg+WHveaVsFxyXG4gIG9uUmVzdWx0VGV4dENoYW5nZWQ6IEZ1bmN0aW9uO1xyXG4gIG9uQmVmb3JlRmluaXNoVGFsazogRnVuY3Rpb247XHJcbiAgb25BZnRlckZpbmlzaFRhbGs6IEZ1bmN0aW9uO1xyXG4gIG9uSW5wdXRDaGFuZ2U6IEZ1bmN0aW9uO1xyXG4gIG9uQmVmb3JlU3RhcnRUYWxrOiBGdW5jdGlvbjtcclxuICBvbkFmdGVyU3RhcnRUYWxrOiBGdW5jdGlvbjtcclxuICBvbkJlZm9yZUNhbmNlbFRhbGs6IEZ1bmN0aW9uO1xyXG4gIG9uQWZ0ZXJDYW5jZWxUYWxrOiBGdW5jdGlvbjtcclxuICBvbkFmdGVyUmVjb3JkU3RhcnQ6IEZ1bmN0aW9uO1xyXG4gIG9uRHVyYXRpb25TdHJDaGFuZ2U6IEZ1bmN0aW9uO1xyXG5cclxuICAvLyBBUEnphY3nva5cclxuICBBUFBJRCA9IFwiNTBmNGE0NmNcIjtcclxuICBBUElfU0VDUkVUID0gXCJOekZsTm1GaFpESmpNRE5rWkdNM056STBNemcyT0dObVwiO1xyXG4gIEFQSV9LRVkgPSBcIjEwNmRkYzQwZGZkNGI5Y2E2ZDdiNDdjNzBmYWRhNzQ5XCI7XHJcblxyXG4gIGNvbnN0cnVjdG9yKFxyXG4gICAgcHJpdmF0ZSBwbGF0Zm9ybTogUGxhdGZvcm0sXHJcbiAgICBwcml2YXRlIGRpYWdub3N0aWM6IERpYWdub3N0aWMsXHJcbiAgKSB7XHJcbiAgICB0aGlzLnJlcXVlc3RQZXJtaXNzaW9uKCk7XHJcbiAgfVxyXG5cclxuICAvLyDnlKjmiLfmk43kvZzmlrnms5VcclxuICB0b2dnbGVSZWNvcmQoKSB7XHJcbiAgICBjb25zb2xlLmxvZygndG9nZ2xlUmVjb3JkJywgdGhpcy5idG5TdGF0dXMpO1xyXG4gICAgaWYgKHRoaXMuYnRuU3RhdHVzID09PSBcIlVOREVGSU5FRFwiIHx8IHRoaXMuYnRuU3RhdHVzID09PSBcIkNMT1NFRFwiKSB7XHJcbiAgICAgIHRoaXMuc3RhcnRUYWxrKCk7XHJcbiAgICB9IGVsc2UgaWYgKHRoaXMuYnRuU3RhdHVzID09PSBcIkNPTk5FQ1RJTkdcIiB8fCB0aGlzLmJ0blN0YXR1cyA9PT0gXCJPUEVOXCIpIHtcclxuICAgICAgdGhpcy5maW5pc2hUYWxrKCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBhc3luYyBzdGFydFRhbGsoZXZlbnQ/KSB7XHJcbiAgICBjb25zb2xlLmxvZygnc3RhcnRUYWxrIGNhbGxlZCcpO1xyXG4gICAgdGhpcy5yZXNldFN0YXRlKCk7XHJcbiAgICB0aGlzLm9uQmVmb3JlU3RhcnRUYWxrICYmIHRoaXMub25CZWZvcmVTdGFydFRhbGsoKTtcclxuICAgIFxyXG4gICAgZXZlbnQ/LnByZXZlbnREZWZhdWx0KCk7XHJcbiAgICBcclxuICAgIHRyeSB7XHJcbiAgICAgIGF3YWl0IHRoaXMub3BlbldpdGhQcml2aWxlZGdlKCk7XHJcbiAgICAgIFxyXG4gICAgICAvLyAxLiDpppblhYjlkK/liqjni6znq4vlvZXliLZcclxuICAgICAgYXdhaXQgdGhpcy5zdGFydEluZGVwZW5kZW50UmVjb3JkaW5nKCk7XHJcbiAgICAgIFxyXG4gICAgICAvLyAyLiDnhLblkI7lkK/liqhXZWJTb2NrZXTov57mjqVcclxuICAgICAgdGhpcy5pc1JlY29yZGluZyA9IHRydWU7XHJcbiAgICAgIHRoaXMuc2hvdWxkUmVjb25uZWN0ID0gdHJ1ZTtcclxuICAgICAgdGhpcy5yZWNvbm5lY3RBdHRlbXB0cyA9IDA7XHJcbiAgICAgIFxyXG4gICAgICBzZXRUaW1lb3V0KCgpID0+IHtcclxuICAgICAgICB0aGlzLmNvbm5lY3RXZWJTb2NrZXQoKTtcclxuICAgICAgfSwgMTAwKTtcclxuXHJcbiAgICAgIHRoaXMuc3RhcnRDb3VudGRvd24oKTtcclxuICAgICAgdGhpcy5vbkFmdGVyU3RhcnRUYWxrICYmIHRoaXMub25BZnRlclN0YXJ0VGFsaygpO1xyXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgY29uc29sZS5lcnJvcignRmFpbGVkIHRvIHN0YXJ0IHRhbGs6JywgZXJyb3IpO1xyXG4gICAgICB0aGlzLnJlc2V0U3RhdGUoKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIGFzeW5jIGZpbmlzaFRhbGsoKSB7XHJcbiAgICBjb25zb2xlLmxvZygnZmluaXNoVGFsayBjYWxsZWQnKTtcclxuICAgIHRoaXMuaXNVc2VyRmluaXNoID0gdHJ1ZTtcclxuICAgIHRoaXMuc2hvdWxkUmVjb25uZWN0ID0gZmFsc2U7XHJcbiAgICB0aGlzLm9uQmVmb3JlRmluaXNoVGFsayAmJiB0aGlzLm9uQmVmb3JlRmluaXNoVGFsaygpO1xyXG4gICAgXHJcbiAgICB0cnkge1xyXG4gICAgICAvLyAxLiDlhYjlhbPpl61XZWJTb2NrZXRcclxuICAgICAgdGhpcy5jbG9zZVdlYlNvY2tldCgpO1xyXG4gICAgICBcclxuICAgICAgLy8gMi4g5YGc5q2i54us56uL5b2V5Yi25bm26I635Y+W5a6M5pW06Z+z6aKRXHJcbiAgICAgIGF3YWl0IHRoaXMuc3RvcEluZGVwZW5kZW50UmVjb3JkaW5nKCk7XHJcbiAgICAgIFxyXG4gICAgICAvLyAzLiDmuIXnkIbnirbmgIFcclxuICAgICAgdGhpcy5pc1JlY29yZGluZyA9IGZhbHNlO1xyXG4gICAgICB0aGlzLmNsZWFyVGltZXJzKCk7XHJcbiAgICAgIHRoaXMuY2hhbmdlQnRuU3RhdHVzKFwiQ0xPU0VEXCIpO1xyXG4gICAgICBcclxuICAgICAgLy8gNC4g5bu26L+f5Zue6LCD56Gu5L+d5omA5pyJ5aSE55CG5a6M5oiQXHJcbiAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xyXG4gICAgICAgIGlmICh0aGlzLmlzVXNlckZpbmlzaCkge1xyXG4gICAgICAgICAgdGhpcy5vbkFmdGVyRmluaXNoVGFsayAmJiB0aGlzLm9uQWZ0ZXJGaW5pc2hUYWxrKCk7XHJcbiAgICAgICAgICB0aGlzLmlzVXNlckZpbmlzaCA9IGZhbHNlO1xyXG4gICAgICAgIH1cclxuICAgICAgfSwgNTAwKTtcclxuICAgICAgXHJcbiAgICB9IGNhdGNoIChlcnJvcikge1xyXG4gICAgICBjb25zb2xlLmVycm9yKCdFcnJvciBmaW5pc2hpbmcgdGFsazonLCBlcnJvcik7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBjYW5jZWxUYWxrKCkge1xyXG4gICAgY29uc29sZS5sb2coJ2NhbmNlbFRhbGsgY2FsbGVkJyk7XHJcbiAgICB0aGlzLm9uQmVmb3JlQ2FuY2VsVGFsayAmJiB0aGlzLm9uQmVmb3JlQ2FuY2VsVGFsaygpO1xyXG4gICAgXHJcbiAgICB0aGlzLmlzVXNlckZpbmlzaCA9IGZhbHNlO1xyXG4gICAgdGhpcy5zaG91bGRSZWNvbm5lY3QgPSBmYWxzZTtcclxuICAgIHRoaXMuaXNSZWNvcmRpbmcgPSBmYWxzZTtcclxuICAgIFxyXG4gICAgLy8g5YWz6ZetV2ViU29ja2V0XHJcbiAgICB0aGlzLmNsb3NlV2ViU29ja2V0KCk7XHJcbiAgICBcclxuICAgIC8vIOWBnOatoueLrOeri+W9leWItuS9huS4jeS/neWtmOe7k+aenFxyXG4gICAgdGhpcy5jYW5jZWxJbmRlcGVuZGVudFJlY29yZGluZygpO1xyXG4gICAgXHJcbiAgICB0aGlzLnJlc2V0U3RhdGUoKTtcclxuICAgIHRoaXMub25BZnRlckNhbmNlbFRhbGsgJiYgdGhpcy5vbkFmdGVyQ2FuY2VsVGFsaygpO1xyXG4gIH1cclxuXHJcbiAgLy8g54us56uL5b2V5Yi25pa55rOVIC0g5LiOV2ViU29ja2V05a6M5YWo6Kej6ICmXHJcbiAgYXN5bmMgc3RhcnRJbmRlcGVuZGVudFJlY29yZGluZygpOiBQcm9taXNlPHZvaWQ+IHtcclxuICAgIGNvbnNvbGUubG9nKCdTdGFydGluZyBpbmRlcGVuZGVudCByZWNvcmRpbmcnKTtcclxuICAgIFxyXG4gICAgdGhpcy5hbGxSZWNvcmRlZEJ1ZmZlcnMgPSBbXTtcclxuICAgIHRoaXMuY3JlYXRlSW5kZXBlbmRlbnRSZWNvcmRlcigpO1xyXG4gICAgXHJcbiAgICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xyXG4gICAgICB0aGlzLmN1cnJlbnRSZWNvcmRlci5vcGVuKCgpID0+IHtcclxuICAgICAgICBjb25zb2xlLmxvZygnSW5kZXBlbmRlbnQgcmVjb3JkZXIgb3BlbmVkJyk7XHJcbiAgICAgICAgXHJcbiAgICAgICAgLy8g5Yib5bu65Y+v6KeG5YyWXHJcbiAgICAgICAgbGV0IHdhdmVEaXYgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yKFwiLnJlY29yZC13YXZlXCIpO1xyXG4gICAgICAgIGlmICh3YXZlRGl2ICYmIFJlY29yZGVyLldhdmVWaWV3KSB7XHJcbiAgICAgICAgICB0aGlzLndhdmVDbGllbnQgPSBSZWNvcmRlci5XYXZlVmlldyh7IGVsZW06IFwiLnJlY29yZC13YXZlXCIgfSk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIFxyXG4gICAgICAgIHRoaXMuY3VycmVudFJlY29yZGVyLnN0YXJ0KCk7XHJcbiAgICAgICAgdGhpcy5vbkFmdGVyUmVjb3JkU3RhcnQgJiYgdGhpcy5vbkFmdGVyUmVjb3JkU3RhcnQoKTtcclxuICAgICAgICByZXNvbHZlKCk7XHJcbiAgICAgIH0sIChtc2csIGlzVXNlck5vdEFsbG93KSA9PiB7XHJcbiAgICAgICAgY29uc29sZS5lcnJvcignRmFpbGVkIHRvIG9wZW4gaW5kZXBlbmRlbnQgcmVjb3JkZXI6JywgbXNnKTtcclxuICAgICAgICByZWplY3QobmV3IEVycm9yKG1zZykpO1xyXG4gICAgICB9KTtcclxuICAgIH0pO1xyXG4gIH1cclxuXHJcbiAgYXN5bmMgc3RvcEluZGVwZW5kZW50UmVjb3JkaW5nKCk6IFByb21pc2U8dm9pZD4ge1xyXG4gICAgY29uc29sZS5sb2coJ1N0b3BwaW5nIGluZGVwZW5kZW50IHJlY29yZGluZycpO1xyXG4gICAgXHJcbiAgICBpZiAoIXRoaXMuY3VycmVudFJlY29yZGVyKSB7XHJcbiAgICAgIGNvbnNvbGUubG9nKCdObyBpbmRlcGVuZGVudCByZWNvcmRlciB0byBzdG9wJyk7XHJcbiAgICAgIHJldHVybjtcclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcclxuICAgICAgdGhpcy5jdXJyZW50UmVjb3JkZXIuc3RvcChcclxuICAgICAgICBhc3luYyAoYmxvYjogQmxvYiwgZHVyYXRpb246IG51bWJlcikgPT4ge1xyXG4gICAgICAgICAgY29uc29sZS5sb2coJ0luZGVwZW5kZW50IHJlY29yZGluZyBzdG9wcGVkIHN1Y2Nlc3NmdWxseScsIGJsb2IsIGR1cmF0aW9uKTtcclxuICAgICAgICAgIFxyXG4gICAgICAgICAgLy8g5L+d5a2Y5a6M5pW055qE5b2V5Yi257uT5p6cXHJcbiAgICAgICAgICB0aGlzLnJlY29yZFBjbUJsb2IgPSBibG9iO1xyXG4gICAgICAgICAgdGhpcy5yZWNvcmRXYXZCbG9iID0gYXdhaXQgdGhpcy5wY21CbG9iVG9XYXZCbG9iKGJsb2IsIDQ0MTAwKTtcclxuICAgICAgICAgIFxyXG4gICAgICAgICAgLy8g5riF55CG5b2V5Yi25ZmoXHJcbiAgICAgICAgICB0aGlzLmNsZWFudXBJbmRlcGVuZGVudFJlY29yZGVyKCk7XHJcbiAgICAgICAgICByZXNvbHZlKCk7XHJcbiAgICAgICAgfSxcclxuICAgICAgICAobXNnKSA9PiB7XHJcbiAgICAgICAgICBjb25zb2xlLmVycm9yKFwi54us56uL5b2V6Z+z5YGc5q2i5aSx6LSlOlwiICsgbXNnKTtcclxuICAgICAgICAgIHRoaXMuY2xlYW51cEluZGVwZW5kZW50UmVjb3JkZXIoKTtcclxuICAgICAgICAgIHJlc29sdmUoKTtcclxuICAgICAgICB9XHJcbiAgICAgICk7XHJcbiAgICB9KTtcclxuICB9XHJcblxyXG4gIGNhbmNlbEluZGVwZW5kZW50UmVjb3JkaW5nKCk6IHZvaWQge1xyXG4gICAgY29uc29sZS5sb2coJ0NhbmNlbGxpbmcgaW5kZXBlbmRlbnQgcmVjb3JkaW5nJyk7XHJcbiAgICBcclxuICAgIGlmICh0aGlzLmN1cnJlbnRSZWNvcmRlcikge1xyXG4gICAgICB0cnkge1xyXG4gICAgICAgIC8vIOebtOaOpeWFs+mXre+8jOS4jeS/neWtmOe7k+aenFxyXG4gICAgICAgIHRoaXMuY3VycmVudFJlY29yZGVyLmNsb3NlKCk7XHJcbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XHJcbiAgICAgICAgY29uc29sZS5lcnJvcignRXJyb3IgY2xvc2luZyBpbmRlcGVuZGVudCByZWNvcmRlcjonLCBlcnJvcik7XHJcbiAgICAgIH1cclxuICAgIH1cclxuICAgIFxyXG4gICAgdGhpcy5jbGVhbnVwSW5kZXBlbmRlbnRSZWNvcmRlcigpO1xyXG4gICAgdGhpcy5yZWNvcmRQY21CbG9iID0gbnVsbDtcclxuICAgIHRoaXMucmVjb3JkV2F2QmxvYiA9IG51bGw7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGNyZWF0ZUluZGVwZW5kZW50UmVjb3JkZXIoKTogdm9pZCB7XHJcbiAgICBpZiAodGhpcy5jdXJyZW50UmVjb3JkZXIpIHtcclxuICAgICAgdGhpcy5jbGVhbnVwSW5kZXBlbmRlbnRSZWNvcmRlcigpO1xyXG4gICAgfVxyXG5cclxuICAgIHRoaXMuY3VycmVudFJlY29yZGVyID0gUmVjb3JkZXIoe1xyXG4gICAgICB0eXBlOiB0aGlzLnJlY29yZFR5cGUsXHJcbiAgICAgIHNhbXBsZVJhdGU6IDQ0MTAwLFxyXG4gICAgICBiaXRSYXRlOiAxNixcclxuICAgICAgb25Qcm9jZXNzOiAoYnVmZmVycywgcG93ZXJMZXZlbCwgYnVmZmVyRHVyYXRpb24sIGJ1ZmZlclNhbXBsZVJhdGUsIG5ld0J1ZmZlcklkeCwgYXN5bmNFbmQpID0+IHtcclxuICAgICAgICBpZiAoIXRoaXMuaXNSZWNvcmRpbmcpIHJldHVybjtcclxuXHJcbiAgICAgICAgLy8g5L+d5a2Y5omA5pyJYnVmZmVy55So5LqO5pyA57uI5ZCI5oiQ5a6M5pW06Z+z6aKRXHJcbiAgICAgICAgdGhpcy5hbGxSZWNvcmRlZEJ1ZmZlcnMgPSBidWZmZXJzO1xyXG5cclxuICAgICAgICAvLyDojrflj5bmnIDmlrDnmoTpn7PpopHniYfmrrXnlKjkuo7lrp7ml7ZBU1JcclxuICAgICAgICBsZXQgZnJhbWVCdWZmZXIgPSBidWZmZXJzLmxlbmd0aCAmJiBidWZmZXJzW2J1ZmZlcnMubGVuZ3RoIC0gMV07XHJcbiAgICAgICAgaWYgKCFmcmFtZUJ1ZmZlcikgcmV0dXJuO1xyXG5cclxuICAgICAgICAvLyDph43ph4fmoLfliLAxNmtIeueUqOS6jkFTUlxyXG4gICAgICAgIGZyYW1lQnVmZmVyID0gcmVzYW1wbGVCdWZmZXIoZnJhbWVCdWZmZXIsIDQ0MTAwLCAxNjAwMCk7XHJcblxyXG4gICAgICAgIC8vIOWPkemAgeWIsFdlYlNvY2tldOi/m+ihjOWunuaXtkFTUlxyXG4gICAgICAgIHRoaXMuc2VuZEF1ZGlvVG9XZWJTb2NrZXQoZnJhbWVCdWZmZXIpO1xyXG5cclxuICAgICAgICAvLyDmm7TmlrDms6LlvaLmmL7npLpcclxuICAgICAgICB0aGlzLndhdmVDbGllbnQ/LmlucHV0KGJ1ZmZlcnNbYnVmZmVycy5sZW5ndGggLSAxXSwgcG93ZXJMZXZlbCwgYnVmZmVyU2FtcGxlUmF0ZSk7XHJcbiAgICAgIH1cclxuICAgIH0pO1xyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBjbGVhbnVwSW5kZXBlbmRlbnRSZWNvcmRlcigpOiB2b2lkIHtcclxuICAgIGlmICh0aGlzLmN1cnJlbnRSZWNvcmRlcikge1xyXG4gICAgICB0cnkge1xyXG4gICAgICAgIHRoaXMuY3VycmVudFJlY29yZGVyLmNsb3NlKCk7XHJcbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XHJcbiAgICAgICAgY29uc29sZS5lcnJvcignRXJyb3IgY2xlYW5pbmcgdXAgaW5kZXBlbmRlbnQgcmVjb3JkZXI6JywgZXJyb3IpO1xyXG4gICAgICB9XHJcbiAgICAgIHRoaXMuY3VycmVudFJlY29yZGVyID0gbnVsbDtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHByaXZhdGUgc2VuZEF1ZGlvVG9XZWJTb2NrZXQoZnJhbWVCdWZmZXI6IGFueSk6IHZvaWQge1xyXG4gICAgaWYgKHRoaXMuaWF0V1MgJiYgdGhpcy5pYXRXUy5yZWFkeVN0YXRlID09PSBXZWJTb2NrZXQuT1BFTiAmJiAhdGhpcy5kaXNhYmxlQVNSKSB7XHJcbiAgICAgIHRyeSB7XHJcbiAgICAgICAgdGhpcy5pYXRXUy5zZW5kKEpTT04uc3RyaW5naWZ5KHtcclxuICAgICAgICAgIGRhdGE6IHtcclxuICAgICAgICAgICAgc3RhdHVzOiAxLFxyXG4gICAgICAgICAgICBmb3JtYXQ6IFwiYXVkaW8vTDE2O3JhdGU9MTYwMDBcIixcclxuICAgICAgICAgICAgZW5jb2Rpbmc6IHRoaXMuZW5jb2RpbmdUeXBlLFxyXG4gICAgICAgICAgICBhdWRpbzogY29udmVydEZyYW1lQnVmZmVyVG9CYXNlNjQoZnJhbWVCdWZmZXIpXHJcbiAgICAgICAgICB9LFxyXG4gICAgICAgIH0pKTtcclxuICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgICBjb25zb2xlLmVycm9yKFwiRXJyb3Igc2VuZGluZyBhdWRpbyBkYXRhIHRvIFdlYlNvY2tldDpcIiwgZXJyb3IpO1xyXG4gICAgICB9XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICAvLyBXZWJTb2NrZXTnm7jlhbPmlrnms5UgLSDku4XnlKjkuo5BU1LvvIzkuI3mjqfliLblvZXliLZcclxuICBjb25uZWN0V2ViU29ja2V0KCkge1xyXG4gICAgY29uc29sZS5sb2coXCJjb25uZWN0V2ViU29ja2V0IGNhbGxlZCwgc2hvdWxkUmVjb25uZWN0OlwiLCB0aGlzLnNob3VsZFJlY29ubmVjdCk7XHJcbiAgICBcclxuICAgIGlmICghdGhpcy5zaG91bGRSZWNvbm5lY3QgfHwgIXRoaXMuaXNSZWNvcmRpbmcpIHtcclxuICAgICAgY29uc29sZS5sb2coXCJOb3QgY29ubmVjdGluZyBXZWJTb2NrZXQgLSBzaG91bGRSZWNvbm5lY3Q6XCIsIHRoaXMuc2hvdWxkUmVjb25uZWN0LCBcImlzUmVjb3JkaW5nOlwiLCB0aGlzLmlzUmVjb3JkaW5nKTtcclxuICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIOWFs+mXreeOsOaciei/nuaOpVxyXG4gICAgdGhpcy5jbG9zZVdlYlNvY2tldCgpO1xyXG5cclxuICAgIGNvbnN0IHdlYnNvY2tldFVybCA9IHRoaXMuZ2V0V2ViU29ja2V0VXJsKCk7XHJcbiAgICBjb25zb2xlLmxvZyhcIkNvbm5lY3RpbmcgdG86XCIsIHdlYnNvY2tldFVybCk7XHJcblxyXG4gICAgdHJ5IHtcclxuICAgICAgaWYgKFwiV2ViU29ja2V0XCIgaW4gd2luZG93KSB7XHJcbiAgICAgICAgdGhpcy5pYXRXUyA9IG5ldyBXZWJTb2NrZXQod2Vic29ja2V0VXJsKTtcclxuICAgICAgfSBlbHNlIHtcclxuICAgICAgICBjb25zb2xlLmVycm9yKFwi5rWP6KeI5Zmo5LiN5pSv5oyBV2ViU29ja2V0XCIpO1xyXG4gICAgICAgIGFsZXJ0KFwi5rWP6KeI5Zmo5LiN5pSv5oyBV2ViU29ja2V0XCIpO1xyXG4gICAgICAgIHJldHVybjtcclxuICAgICAgfVxyXG5cclxuICAgICAgdGhpcy5jaGFuZ2VCdG5TdGF0dXMoXCJDT05ORUNUSU5HXCIpO1xyXG5cclxuICAgICAgdGhpcy5pYXRXUy5vbm9wZW4gPSAoZSkgPT4ge1xyXG4gICAgICAgIGNvbnNvbGUubG9nKFwiV2ViU29ja2V0IGNvbm5lY3RlZCBmb3IgQVNSXCIpO1xyXG4gICAgICAgIHRoaXMucmVjb25uZWN0QXR0ZW1wdHMgPSAwO1xyXG4gICAgICAgIHRoaXMuY2hhbmdlQnRuU3RhdHVzKFwiT1BFTlwiKTtcclxuXHJcbiAgICAgICAgLy8g5Y+R6YCB5Yid5aeL5Y+C5pWwXHJcbiAgICAgICAgY29uc3QgcGFyYW1zID0ge1xyXG4gICAgICAgICAgY29tbW9uOiB7XHJcbiAgICAgICAgICAgIGFwcF9pZDogdGhpcy5BUFBJRCxcclxuICAgICAgICAgIH0sXHJcbiAgICAgICAgICBidXNpbmVzczoge1xyXG4gICAgICAgICAgICBsYW5ndWFnZTogXCJ6aF9jblwiLFxyXG4gICAgICAgICAgICBkb21haW46IFwiaWF0XCIsXHJcbiAgICAgICAgICAgIGFjY2VudDogXCJtYW5kYXJpblwiLFxyXG4gICAgICAgICAgICB2YWRfZW9zOiAxMDAwMCwgLy8g5aKe5Yqg5YiwMTDnp5JcclxuICAgICAgICAgICAgZHdhOiBcIndwZ3NcIixcclxuICAgICAgICAgIH0sXHJcbiAgICAgICAgICBkYXRhOiB7XHJcbiAgICAgICAgICAgIHN0YXR1czogMCxcclxuICAgICAgICAgICAgZm9ybWF0OiBcImF1ZGlvL0wxNjtyYXRlPTE2MDAwXCIsXHJcbiAgICAgICAgICAgIGVuY29kaW5nOiB0aGlzLmVuY29kaW5nVHlwZSxcclxuICAgICAgICAgIH0sXHJcbiAgICAgICAgfTtcclxuICAgICAgICB0aGlzLmlhdFdTLnNlbmQoSlNPTi5zdHJpbmdpZnkocGFyYW1zKSk7XHJcbiAgICAgIH07XHJcblxyXG4gICAgICB0aGlzLmlhdFdTLm9ubWVzc2FnZSA9IChlKSA9PiB7XHJcbiAgICAgICAgdGhpcy5yZW5kZXJSZXN1bHQoZS5kYXRhKTtcclxuICAgICAgfTtcclxuXHJcbiAgICAgIHRoaXMuaWF0V1Mub25lcnJvciA9IChlKSA9PiB7XHJcbiAgICAgICAgY29uc29sZS5lcnJvcihcIldlYlNvY2tldCBlcnJvcjpcIiwgZSk7XHJcbiAgICAgICAgdGhpcy5oYW5kbGVXZWJTb2NrZXRFcnJvcigpO1xyXG4gICAgICB9O1xyXG5cclxuICAgICAgdGhpcy5pYXRXUy5vbmNsb3NlID0gKGUpID0+IHtcclxuICAgICAgICBjb25zb2xlLmxvZyhcIldlYlNvY2tldCBjbG9zZWQ6XCIsIGUuY29kZSwgZS5yZWFzb24pO1xyXG4gICAgICAgIHRoaXMuaGFuZGxlV2ViU29ja2V0Q2xvc2UoZSk7XHJcbiAgICAgIH07XHJcblxyXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgY29uc29sZS5lcnJvcihcIkZhaWxlZCB0byBjcmVhdGUgV2ViU29ja2V0OlwiLCBlcnJvcik7XHJcbiAgICAgIHRoaXMuaGFuZGxlV2ViU29ja2V0RXJyb3IoKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHByaXZhdGUgaGFuZGxlV2ViU29ja2V0Q2xvc2UoZXZlbnQ6IENsb3NlRXZlbnQpIHtcclxuICAgIGNvbnNvbGUubG9nKFwiaGFuZGxlV2ViU29ja2V0Q2xvc2VcIiwge1xyXG4gICAgICBjb2RlOiBldmVudC5jb2RlLFxyXG4gICAgICByZWFzb246IGV2ZW50LnJlYXNvbixcclxuICAgICAgc2hvdWxkUmVjb25uZWN0OiB0aGlzLnNob3VsZFJlY29ubmVjdCxcclxuICAgICAgaXNSZWNvcmRpbmc6IHRoaXMuaXNSZWNvcmRpbmcsXHJcbiAgICAgIGlzVXNlckZpbmlzaDogdGhpcy5pc1VzZXJGaW5pc2hcclxuICAgIH0pO1xyXG5cclxuICAgIC8vIOWmguaenOeUqOaIt+W3sue7j+e7k+adn+aIluS4jeW6lOivpemHjei/nu+8jOWImeS4jeWkhOeQhlxyXG4gICAgaWYgKHRoaXMuaXNVc2VyRmluaXNoIHx8ICF0aGlzLnNob3VsZFJlY29ubmVjdCB8fCAhdGhpcy5pc1JlY29yZGluZykge1xyXG4gICAgICByZXR1cm47XHJcbiAgICB9XHJcblxyXG4gICAgLy8g5Y+R6YCB57uT5p2f5bin77yI5aaC5p6c6ZyA6KaB77yJXHJcbiAgICB0cnkge1xyXG4gICAgICBpZiAodGhpcy5pYXRXUyAmJiB0aGlzLmlhdFdTLnJlYWR5U3RhdGUgPT09IFdlYlNvY2tldC5PUEVOKSB7XHJcbiAgICAgICAgdGhpcy5pYXRXUy5zZW5kKEpTT04uc3RyaW5naWZ5KHtcclxuICAgICAgICAgIFwiZGF0YVwiOiB7XHJcbiAgICAgICAgICAgIFwic3RhdHVzXCI6IDIsXHJcbiAgICAgICAgICAgIFwiZm9ybWF0XCI6IFwiYXVkaW8vTDE2O3JhdGU9MTYwMDBcIixcclxuICAgICAgICAgICAgXCJlbmNvZGluZ1wiOiB0aGlzLmVuY29kaW5nVHlwZSxcclxuICAgICAgICAgICAgXCJhdWRpb1wiOiBcIlwiXHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgfSkpO1xyXG4gICAgICB9XHJcbiAgICB9IGNhdGNoIChlcnIpIHtcclxuICAgICAgY29uc29sZS5lcnJvcignRXJyb3Igc2VuZGluZyBlbmQgZnJhbWUgb24gY2xvc2U6JywgZXJyKTtcclxuICAgIH1cclxuXHJcbiAgICAvLyDmraPluLjlhbPpl63noIHmiJbmnI3liqHnq6/kuLvliqjlhbPpl63vvIzlsJ3or5Xph43ov55cclxuICAgIGlmIChldmVudC5jb2RlID09PSAxMDAwIHx8IGV2ZW50LmNvZGUgPT09IDEwMDYgfHwgZXZlbnQuY29kZSA9PT0gMTAxMSkge1xyXG4gICAgICB0aGlzLmF0dGVtcHRSZWNvbm5lY3QoKTtcclxuICAgIH0gZWxzZSB7XHJcbiAgICAgIGNvbnNvbGUuZXJyb3IoXCJXZWJTb2NrZXQgY2xvc2VkIHdpdGggZXJyb3IgY29kZTpcIiwgZXZlbnQuY29kZSk7XHJcbiAgICAgIHRoaXMuaGFuZGxlV2ViU29ja2V0RXJyb3IoKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHByaXZhdGUgaGFuZGxlV2ViU29ja2V0RXJyb3IoKSB7XHJcbiAgICBpZiAoIXRoaXMuc2hvdWxkUmVjb25uZWN0IHx8ICF0aGlzLmlzUmVjb3JkaW5nKSB7XHJcbiAgICAgIHJldHVybjtcclxuICAgIH1cclxuICAgIHRoaXMuYXR0ZW1wdFJlY29ubmVjdCgpO1xyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBhdHRlbXB0UmVjb25uZWN0KCkge1xyXG4gICAgaWYgKHRoaXMucmVjb25uZWN0QXR0ZW1wdHMgPj0gdGhpcy5tYXhSZWNvbm5lY3RBdHRlbXB0cykge1xyXG4gICAgICBjb25zb2xlLmVycm9yKFwiTWF4IHJlY29ubmVjdCBhdHRlbXB0cyByZWFjaGVkXCIpO1xyXG4gICAgICAvLyDkuI3mlLnlj5jlvZXliLbnirbmgIHvvIzlj6rmmK9BU1LmmoLml7bkuI3lj6/nlKhcclxuICAgICAgcmV0dXJuO1xyXG4gICAgfVxyXG5cclxuICAgIHRoaXMucmVjb25uZWN0QXR0ZW1wdHMrKztcclxuICAgIGNvbnNvbGUubG9nKGBBdHRlbXB0aW5nIFdlYlNvY2tldCByZWNvbm5lY3QgJHt0aGlzLnJlY29ubmVjdEF0dGVtcHRzfS8ke3RoaXMubWF4UmVjb25uZWN0QXR0ZW1wdHN9YCk7XHJcblxyXG4gICAgc2V0VGltZW91dCgoKSA9PiB7XHJcbiAgICAgIGlmICh0aGlzLnNob3VsZFJlY29ubmVjdCAmJiB0aGlzLmlzUmVjb3JkaW5nICYmICF0aGlzLmlzVXNlckZpbmlzaCkge1xyXG4gICAgICAgIHRoaXMuY29ubmVjdFdlYlNvY2tldCgpO1xyXG4gICAgICB9XHJcbiAgICB9LCB0aGlzLnJlY29ubmVjdERlbGF5ICogdGhpcy5yZWNvbm5lY3RBdHRlbXB0cyk7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGNsb3NlV2ViU29ja2V0KCkge1xyXG4gICAgaWYgKHRoaXMuaWF0V1MpIHtcclxuICAgICAgdHJ5IHtcclxuICAgICAgICAvLyDlj5HpgIHnu5PmnZ/luKdcclxuICAgICAgICBpZiAodGhpcy5pYXRXUy5yZWFkeVN0YXRlID09PSBXZWJTb2NrZXQuT1BFTikge1xyXG4gICAgICAgICAgdGhpcy5pYXRXUy5zZW5kKEpTT04uc3RyaW5naWZ5KHtcclxuICAgICAgICAgICAgXCJkYXRhXCI6IHtcclxuICAgICAgICAgICAgICBcInN0YXR1c1wiOiAyLFxyXG4gICAgICAgICAgICAgIFwiZm9ybWF0XCI6IFwiYXVkaW8vTDE2O3JhdGU9MTYwMDBcIixcclxuICAgICAgICAgICAgICBcImVuY29kaW5nXCI6IHRoaXMuZW5jb2RpbmdUeXBlLFxyXG4gICAgICAgICAgICAgIFwiYXVkaW9cIjogXCJcIlxyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICB9KSk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIFxyXG4gICAgICAgIGlmICh0aGlzLmlhdFdTLnJlYWR5U3RhdGUgPT09IFdlYlNvY2tldC5PUEVOIHx8IHRoaXMuaWF0V1MucmVhZHlTdGF0ZSA9PT0gV2ViU29ja2V0LkNPTk5FQ1RJTkcpIHtcclxuICAgICAgICAgIHRoaXMuaWF0V1MuY2xvc2UoMTAwMCwgXCJVc2VyIGluaXRpYXRlZCBjbG9zZVwiKTtcclxuICAgICAgICB9XHJcbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XHJcbiAgICAgICAgY29uc29sZS5lcnJvcihcIkVycm9yIGNsb3NpbmcgV2ViU29ja2V0OlwiLCBlcnJvcik7XHJcbiAgICAgIH1cclxuICAgICAgdGhpcy5pYXRXUyA9IG51bGw7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICAvLyDnirbmgIHnrqHnkIZcclxuICBwcml2YXRlIHJlc2V0U3RhdGUoKSB7XHJcbiAgICB0aGlzLnJlc3VsdFRleHQgPSBcIlwiO1xyXG4gICAgdGhpcy5yZXN1bHRUZXh0VGVtcCA9IFwiXCI7XHJcbiAgICB0aGlzLmR1cmF0aW9uU3RyID0gXCIwMDowMFwiO1xyXG4gICAgdGhpcy5kdXJhdGlvbiA9IDA7XHJcbiAgICB0aGlzLnJlY29yZER1cmF0aW9uID0gMDtcclxuICAgIHRoaXMuYWxsUmVjb3JkZWRCdWZmZXJzID0gW107XHJcbiAgICB0aGlzLmNsZWFyVGltZXJzKCk7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGNsZWFyVGltZXJzKCkge1xyXG4gICAgaWYgKHRoaXMuY291bnRkb3duSW50ZXJ2YWwpIHtcclxuICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLmNvdW50ZG93bkludGVydmFsKTtcclxuICAgICAgdGhpcy5jb3VudGRvd25JbnRlcnZhbCA9IG51bGw7XHJcbiAgICB9XHJcbiAgICBpZiAodGhpcy5kdXJhdGlvbkludGVydmFsKSB7XHJcbiAgICAgIGNsZWFySW50ZXJ2YWwodGhpcy5kdXJhdGlvbkludGVydmFsKTtcclxuICAgICAgdGhpcy5kdXJhdGlvbkludGVydmFsID0gbnVsbDtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIGNoYW5nZUJ0blN0YXR1cyhzdGF0dXM6IHN0cmluZykge1xyXG4gICAgdGhpcy5idG5TdGF0dXMgPSBzdGF0dXM7XHJcbiAgICBzd2l0Y2ggKHN0YXR1cykge1xyXG4gICAgICBjYXNlIFwiQ09OTkVDVElOR1wiOlxyXG4gICAgICAgIHRoaXMuY29ublN0YXR1cyA9IFwi5bu656uL6L+e5o6l5LitXCI7XHJcbiAgICAgICAgYnJlYWs7XHJcbiAgICAgIGNhc2UgXCJPUEVOXCI6XHJcbiAgICAgICAgdGhpcy5jb25uU3RhdHVzID0gYOW9lemfs+S4re+8iCR7dGhpcy5kdXJhdGlvblN0cn3vvIlgO1xyXG4gICAgICAgIGJyZWFrO1xyXG4gICAgICBjYXNlIFwiQ0xPU0lOR1wiOlxyXG4gICAgICAgIHRoaXMuY29ublN0YXR1cyA9IFwi5YWz6Zet6L+e5o6l5LitXCI7XHJcbiAgICAgICAgYnJlYWs7XHJcbiAgICAgIGNhc2UgXCJDTE9TRURcIjpcclxuICAgICAgICB0aGlzLmNvbm5TdGF0dXMgPSBcIuW8gOWni+W9lemfs1wiO1xyXG4gICAgICAgIGJyZWFrO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgLy8g6K6h5pe25ZmoXHJcbiAgc3RhcnRDb3VudGRvd24oKSB7XHJcbiAgICB0aGlzLmNsZWFyVGltZXJzKCk7XHJcbiAgICB0aGlzLnJlY29yZER1cmF0aW9uID0gMDtcclxuICAgIHRoaXMuZHVyYXRpb24gPSAwO1xyXG4gICAgdGhpcy5ub3cgPSBuZXcgRGF0ZSgpO1xyXG5cclxuICAgIC8vIOW9lemfs+aXtumVv+iuoeaXtuWZqO+8iDEwMG1z57K+5bqm77yJXHJcbiAgICB0aGlzLmR1cmF0aW9uSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbCgoKSA9PiB7XHJcbiAgICAgIGlmICh0aGlzLmlzUmVjb3JkaW5nKSB7XHJcbiAgICAgICAgdGhpcy5yZWNvcmREdXJhdGlvbiArPSAxMDA7XHJcbiAgICAgIH1cclxuICAgIH0sIDEwMCk7XHJcblxyXG4gICAgLy8g5pi+56S65pe26ZW/6K6h5pe25Zmo77yIMeenkueyvuW6pu+8iVxyXG4gICAgdGhpcy5jb3VudGRvd25JbnRlcnZhbCA9IHNldEludGVydmFsKCgpID0+IHtcclxuICAgICAgaWYgKHRoaXMuaXNSZWNvcmRpbmcpIHtcclxuICAgICAgICB0aGlzLmNvdW50VGltZXIoKTtcclxuICAgICAgfVxyXG4gICAgfSwgMTAwMCk7XHJcbiAgfVxyXG5cclxuICBjb3VudFRpbWVyKCkge1xyXG4gICAgdGhpcy5kdXJhdGlvbisrO1xyXG4gICAgbGV0IG1pbnV0ZVN0ciA9IFN0cmluZyhNYXRoLmZsb29yKHRoaXMuZHVyYXRpb24gLyA2MCkpLnBhZFN0YXJ0KDIsIFwiMFwiKTtcclxuICAgIGxldCBzZWNvbmRTdHIgPSBTdHJpbmcodGhpcy5kdXJhdGlvbiAlIDYwKS5wYWRTdGFydCgyLCBcIjBcIik7XHJcbiAgICBsZXQgZHVyYXRpb25TdHIgPSBtaW51dGVTdHIgKyBcIjpcIiArIHNlY29uZFN0cjtcclxuICAgIHRoaXMuZHVyYXRpb25TdHIgPSBkdXJhdGlvblN0cjtcclxuICAgIHRoaXMuY29ublN0YXR1cyA9IGDlvZXpn7PkuK3vvIgke3RoaXMuZHVyYXRpb25TdHJ977yJYDtcclxuICAgIHRoaXMub25EdXJhdGlvblN0ckNoYW5nZSAmJiB0aGlzLm9uRHVyYXRpb25TdHJDaGFuZ2UoZHVyYXRpb25TdHIpO1xyXG4gIH1cclxuXHJcbiAgLy8g57uT5p6c5aSE55CGXHJcbiAgcmVuZGVyUmVzdWx0KHJlc3VsdERhdGE6IHN0cmluZykge1xyXG4gICAgdHJ5IHtcclxuICAgICAgbGV0IGpzb25EYXRhID0gSlNPTi5wYXJzZShyZXN1bHREYXRhKTtcclxuICAgICAgXHJcbiAgICAgIGlmIChqc29uRGF0YS5kYXRhICYmIGpzb25EYXRhLmRhdGEucmVzdWx0KSB7XHJcbiAgICAgICAgbGV0IGRhdGEgPSBqc29uRGF0YS5kYXRhLnJlc3VsdDtcclxuICAgICAgICBsZXQgc3RyID0gXCJcIjtcclxuICAgICAgICBsZXQgd3MgPSBkYXRhLndzO1xyXG4gICAgICAgIFxyXG4gICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgd3MubGVuZ3RoOyBpKyspIHtcclxuICAgICAgICAgIHN0ciA9IHN0ciArIHdzW2ldLmN3WzBdLnc7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICAvLyDlpITnkIbliqjmgIHkv67mraNcclxuICAgICAgICBpZiAoZGF0YS5wZ3MpIHtcclxuICAgICAgICAgIGlmIChkYXRhLnBncyA9PT0gXCJhcGRcIikge1xyXG4gICAgICAgICAgICB0aGlzLnJlc3VsdFRleHQgPSB0aGlzLnJlc3VsdFRleHRUZW1wO1xyXG4gICAgICAgICAgfVxyXG4gICAgICAgICAgdGhpcy5yZXN1bHRUZXh0VGVtcCA9IHRoaXMucmVzdWx0VGV4dCArIHN0cjtcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgdGhpcy5yZXN1bHRUZXh0ID0gdGhpcy5yZXN1bHRUZXh0ICsgc3RyO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgdGhpcy5vbklucHV0Q2hhbmdlICYmIHRoaXMub25JbnB1dENoYW5nZSh0aGlzLmdldFVzZXJJbnB1dCgpKTtcclxuICAgICAgfVxyXG5cclxuICAgICAgLy8g5aSE55CG57uT5p2f54q25oCBXHJcbiAgICAgIGlmIChqc29uRGF0YS5jb2RlID09PSAwICYmIGpzb25EYXRhLmRhdGEuc3RhdHVzID09PSAyKSB7XHJcbiAgICAgICAgY29uc29sZS5sb2coXCJBU1Igc2Vzc2lvbiBjb21wbGV0ZWQsIHdpbGwgcmVjb25uZWN0IGlmIHN0aWxsIHJlY29yZGluZ1wiKTtcclxuICAgICAgICBpZiAodGhpcy5pYXRXUyAmJiB0aGlzLmlhdFdTLnJlYWR5U3RhdGUgPT09IFdlYlNvY2tldC5PUEVOKSB7XHJcbiAgICAgICAgICB0aGlzLmlhdFdTLmNsb3NlKCk7XHJcbiAgICAgICAgfVxyXG4gICAgICB9XHJcblxyXG4gICAgICAvLyDlpITnkIbplJnor69cclxuICAgICAgaWYgKGpzb25EYXRhLmNvZGUgIT09IDApIHtcclxuICAgICAgICBjb25zb2xlLmVycm9yKFwiQVNSIGVycm9yOlwiLCBqc29uRGF0YSk7XHJcbiAgICAgICAgaWYgKHRoaXMuaWF0V1MgJiYgdGhpcy5pYXRXUy5yZWFkeVN0YXRlID09PSBXZWJTb2NrZXQuT1BFTikge1xyXG4gICAgICAgICAgdGhpcy5pYXRXUy5jbG9zZSgpO1xyXG4gICAgICAgIH1cclxuICAgICAgfVxyXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgY29uc29sZS5lcnJvcihcIkVycm9yIHBhcnNpbmcgcmVzdWx0OlwiLCBlcnJvcik7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBnZXRVc2VySW5wdXQoKTogc3RyaW5nIHtcclxuICAgIHJldHVybiBcIlwiICsgKHRoaXMucmVzdWx0VGV4dFRlbXAgfHwgdGhpcy5yZXN1bHRUZXh0KTtcclxuICB9XHJcblxyXG4gIC8vIOadg+mZkOWSjOWIneWni+WMllxyXG4gIGFzeW5jIG9wZW5XaXRoUHJpdmlsZWRnZSgpOiBQcm9taXNlPGJvb2xlYW4+IHtcclxuICAgIGF3YWl0IHRoaXMucmVxdWVzdFBlcm1pc3Npb24oKTtcclxuICAgIFxyXG4gICAgaWYgKFJlY29yZGVyLklzT3BlbigpKSByZXR1cm4gdHJ1ZTtcclxuXHJcbiAgICAvLyDov5nph4zkuI3liJvlu7rlvZXliLblmajvvIzlm6DkuLrmiJHku6zkvb/nlKjni6znq4vnmoTlvZXliLblmahcclxuICAgIHJldHVybiB0cnVlO1xyXG4gIH1cclxuXHJcbiAgLy8g6Z+z6aKR5aSE55CG5pa55rOVXHJcbiAgYXN5bmMgcGNtQmxvYlRvV2F2QmxvYihwY21CbG9iOiBCbG9iLCBzYW1wbGVSYXRlOiBudW1iZXIpOiBQcm9taXNlPEJsb2I+IHtcclxuICAgIHJldHVybiBuZXcgUHJvbWlzZShyZXNvbHZlID0+IHtcclxuICAgICAgbGV0IGZpbGVSZWFkZXIgPSBuZXcgRmlsZVJlYWRlcigpO1xyXG4gICAgICBcclxuICAgICAgZmlsZVJlYWRlci5vbmxvYWQgPSBmdW5jdGlvbihldmVudCkge1xyXG4gICAgICAgIGxldCBwY21EYXRhOiBhbnkgPSBldmVudC50YXJnZXQucmVzdWx0O1xyXG4gICAgICAgIGxldCB3YXZCbG9iID0gcGNtdG9XYXYocGNtRGF0YSwgc2FtcGxlUmF0ZSwgMSwgMTYpO1xyXG4gICAgICAgIHJlc29sdmUod2F2QmxvYik7XHJcbiAgICAgIH07XHJcbiAgICAgIFxyXG4gICAgICBmaWxlUmVhZGVyLnJlYWRBc0FycmF5QnVmZmVyKHBjbUJsb2IpO1xyXG4gICAgfSk7XHJcbiAgfVxyXG5cclxuICBhc3luYyBwbGF5UENNKHBjbUJsb2I6IEJsb2IsIHNhbXBsZVJhdGU6IG51bWJlcikge1xyXG4gICAgbGV0IHdhdkJsb2IgPSBhd2FpdCB0aGlzLnBjbUJsb2JUb1dhdkJsb2IocGNtQmxvYiwgc2FtcGxlUmF0ZSk7XHJcbiAgICBsZXQgd2F2VXJsID0gd2luZG93LlVSTC5jcmVhdGVPYmplY3RVUkwod2F2QmxvYik7XHJcbiAgICBsZXQgYXVkaW8gPSBuZXcgQXVkaW8oKTtcclxuICAgIGF1ZGlvLnNyYyA9IHdhdlVybDtcclxuICAgIGF1ZGlvLnBsYXkoKTtcclxuICB9XHJcblxyXG4gIHBsYXlSZWNvcmQoKSB7XHJcbiAgICBpZiAodGhpcy5yZWNvcmRQY21CbG9iKSB7XHJcbiAgICAgIHRoaXMucGxheVBDTSh0aGlzLnJlY29yZFBjbUJsb2IsIDQ0MTAwKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIC8vIFdlYlNvY2tldCBVUkznlJ/miJBcclxuICBnZXRXZWJTb2NrZXRVcmwoKTogc3RyaW5nIHtcclxuICAgIGxldCB1cmwgPSBcIndzczovL2lhdC1hcGkueGZ5dW4uY24vdjIvaWF0XCI7XHJcbiAgICBsZXQgaG9zdCA9IFwiaWF0LWFwaS54Znl1bi5jblwiO1xyXG4gICAgXHJcbiAgICBsZXQgYXBpS2V5ID0gdGhpcy5BUElfS0VZO1xyXG4gICAgbGV0IGFwaVNlY3JldCA9IHRoaXMuQVBJX1NFQ1JFVDtcclxuICAgIGxldCBkYXRlID0gbmV3IERhdGUoKS50b1VUQ1N0cmluZygpO1xyXG4gICAgbGV0IGFsZ29yaXRobSA9IFwiaG1hYy1zaGEyNTZcIjtcclxuICAgIGxldCBoZWFkZXJzID0gXCJob3N0IGRhdGUgcmVxdWVzdC1saW5lXCI7XHJcbiAgICBsZXQgc2lnbmF0dXJlT3JpZ2luID0gYGhvc3Q6ICR7aG9zdH1cXG5kYXRlOiAke2RhdGV9XFxuR0VUIC92Mi9pYXQgSFRUUC8xLjFgO1xyXG4gICAgbGV0IHNpZ25hdHVyZVNoYSA9IENyeXB0b0pTLkhtYWNTSEEyNTYoc2lnbmF0dXJlT3JpZ2luLCBhcGlTZWNyZXQpO1xyXG4gICAgbGV0IHNpZ25hdHVyZSA9IENyeXB0b0pTLmVuYy5CYXNlNjQuc3RyaW5naWZ5KHNpZ25hdHVyZVNoYSk7XHJcbiAgICBsZXQgYXV0aG9yaXphdGlvbk9yaWdpbiA9IGBhcGlfa2V5PVwiJHthcGlLZXl9XCIsIGFsZ29yaXRobT1cIiR7YWxnb3JpdGhtfVwiLCBoZWFkZXJzPVwiJHtoZWFkZXJzfVwiLCBzaWduYXR1cmU9XCIke3NpZ25hdHVyZX1cImA7XHJcbiAgICBsZXQgYXV0aG9yaXphdGlvbiA9IGJ0b2EoYXV0aG9yaXphdGlvbk9yaWdpbik7XHJcbiAgICB1cmwgPSBgJHt1cmx9P2F1dGhvcml6YXRpb249JHthdXRob3JpemF0aW9ufSZkYXRlPSR7ZGF0ZX0maG9zdD0ke2hvc3R9YDtcclxuICAgIHJldHVybiB1cmw7XHJcbiAgfVxyXG5cclxuICB0b0Jhc2U2NChidWZmZXI6IEFycmF5QnVmZmVyKTogc3RyaW5nIHtcclxuICAgIHZhciBiaW5hcnkgPSBcIlwiO1xyXG4gICAgdmFyIGJ5dGVzID0gbmV3IFVpbnQ4QXJyYXkoYnVmZmVyKTtcclxuICAgIHZhciBsZW4gPSBieXRlcy5ieXRlTGVuZ3RoO1xyXG4gICAgZm9yICh2YXIgaSA9IDA7IGkgPCBsZW47IGkrKykge1xyXG4gICAgICBiaW5hcnkgKz0gU3RyaW5nLmZyb21DaGFyQ29kZShieXRlc1tpXSk7XHJcbiAgICB9XHJcbiAgICByZXR1cm4gd2luZG93LmJ0b2EoYmluYXJ5KTtcclxuICB9XHJcblxyXG4gIC8vIOenu+WKqOerr+adg+mZkOaWueazlVxyXG4gIGlzQ2FwYWNpdG9yKCk6IGJvb2xlYW4ge1xyXG4gICAgaWYoIXRoaXMucGxhdGZvcm0/LmlzKSByZXR1cm4gZmFsc2VcclxuICAgIHJldHVybiB0aGlzLnBsYXRmb3JtLmlzKFwiY2FwYWNpdG9yXCIpIHx8IHRoaXMucGxhdGZvcm0uaXMoXCJjb3Jkb3ZhXCIpO1xyXG4gIH1cclxuXHJcbiAgYXN5bmMgcmVxdWVzdFBlcm1pc3Npb24oKSB7XHJcbiAgICBpZiAoIXRoaXMuaXNDYXBhY2l0b3IoKSkgcmV0dXJuO1xyXG4gICAgXHJcbiAgICB0cnkge1xyXG4gICAgICBhd2FpdCB0aGlzLnJlcXVlc3RTdG9hZ2VQZXJtaXNzaW9uKCk7XHJcbiAgICAgIGF3YWl0IHRoaXMucmVxdWVzdENhbWVyYVBlcm1pc3Npb24oKTtcclxuICAgICAgYXdhaXQgdGhpcy5yZXF1ZXN0TWljUGVybWlzc2lvbigpO1xyXG4gICAgICBhd2FpdCB0aGlzLnJlcXVlc3RSZWNvcmRBdWRpb1Blcm1pc3Npb24oKTtcclxuICAgIH0gY2F0Y2ggKGVycikge1xyXG4gICAgICBjb25zb2xlLmVycm9yKGVycik7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBhc3luYyByZXF1ZXN0UmVjb3JkQXVkaW9QZXJtaXNzaW9uKCkge2xldCBkYXRhID0gYXdhaXQgdGhpcy5kaWFnbm9zdGljLnJlcXVlc3RSdW50aW1lUGVybWlzc2lvbnMoW3RoaXMuZGlhZ25vc3RpYy5wZXJtaXNzaW9uLlJFQ09SRF9BVURJT10pO1xyXG4gICAgY29uc29sZS5sb2coXCJyZWNvcmQgcGVybWlzc2lvbiByZXF1ZXN0OlwiLCBkYXRhKTtcclxuICAgIHJldHVybjtcclxuICB9XHJcblxyXG4gIGFzeW5jIHJlcXVlc3RNaWNQZXJtaXNzaW9uKCkge1xyXG4gICAgbGV0IGlzQXZhaWxhYmxlID0gYXdhaXQgdGhpcy5kaWFnbm9zdGljLmlzTWljcm9waG9uZUF1dGhvcml6ZWQoKTtcclxuICAgIGNvbnNvbGUubG9nKFwicGVybWlzc29uX01JQzpcIiwgaXNBdmFpbGFibGUpO1xyXG4gICAgaWYgKCFpc0F2YWlsYWJsZSkge1xyXG4gICAgICBsZXQgZGF0YSA9IGF3YWl0IHRoaXMuZGlhZ25vc3RpYy5yZXF1ZXN0TWljcm9waG9uZUF1dGhvcml6YXRpb24oKTtcclxuICAgIH1cclxuICAgIHJldHVybjtcclxuICB9XHJcblxyXG4gIGFzeW5jIHJlcXVlc3RTdG9hZ2VQZXJtaXNzaW9uKCkge1xyXG4gICAgbGV0IGlzQXZhaWxhYmxlID0gYXdhaXQgdGhpcy5kaWFnbm9zdGljLmlzRXh0ZXJuYWxTdG9yYWdlQXV0aG9yaXplZCgpO1xyXG4gICAgY29uc29sZS5sb2coXCJwZXJtaXNzb25fU1RPUkFHRTpcIiwgaXNBdmFpbGFibGUpO1xyXG4gICAgaWYgKCFpc0F2YWlsYWJsZSkge1xyXG4gICAgICBsZXQgZGF0YSA9IGF3YWl0IHRoaXMuZGlhZ25vc3RpYy5yZXF1ZXN0RXh0ZXJuYWxTdG9yYWdlQXV0aG9yaXphdGlvbigpO1xyXG4gICAgfVxyXG4gICAgcmV0dXJuO1xyXG4gIH1cclxuXHJcbiAgYXN5bmMgcmVxdWVzdENhbWVyYVBlcm1pc3Npb24oKSB7XHJcbiAgICBsZXQgaXNBdmFpbGFibGUgPSBhd2FpdCB0aGlzLmRpYWdub3N0aWMuaXNDYW1lcmFBdXRob3JpemVkKCk7XHJcbiAgICBjb25zb2xlLmxvZyhcInBlcm1pc3Nvbl9DYW1lcmE6XCIsIGlzQXZhaWxhYmxlKTtcclxuICAgIGlmICghaXNBdmFpbGFibGUpIHtcclxuICAgICAgbGV0IGRhdGEgPSBhd2FpdCB0aGlzLmRpYWdub3N0aWMucmVxdWVzdENhbWVyYUF1dGhvcml6YXRpb24oKTtcclxuICAgIH1cclxuICAgIHJldHVybjtcclxuICB9XHJcblxyXG4gIC8vIOWFtuS7lui+heWKqeaWueazlVxyXG4gIHNwbGl0QXVkaW9EYXRhKGF1ZGlvRGF0YTogYW55KSB7XHJcbiAgICBjb25zdCBzZWdtZW50U2l6ZSA9IDEyODA7XHJcbiAgICBjb25zdCBzZWdtZW50Q291bnQgPSBNYXRoLmNlaWwoYXVkaW9EYXRhLmxlbmd0aCAvIHNlZ21lbnRTaXplKTtcclxuICAgIGNvbnN0IHNlZ21lbnRzID0gW107XHJcblxyXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBzZWdtZW50Q291bnQ7IGkrKykge1xyXG4gICAgICBjb25zdCBzdGFydCA9IGkgKiBzZWdtZW50U2l6ZTtcclxuICAgICAgY29uc3QgZW5kID0gc3RhcnQgKyBzZWdtZW50U2l6ZTtcclxuICAgICAgY29uc3Qgc2VnbWVudCA9IGF1ZGlvRGF0YS5zbGljZShzdGFydCwgZW5kKTtcclxuICAgICAgc2VnbWVudHMucHVzaChzZWdtZW50KTtcclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4gc2VnbWVudHM7XHJcbiAgfVxyXG5cclxuICBCdWZmZXJUb0Jsb2IoYnVmZmVyOiBhbnkpIHtcclxuICAgIHJldHVybiBuZXcgQmxvYihbYnVmZmVyXSwgeyB0eXBlOiAnYXVkaW8vcGNtJyB9KTtcclxuICB9XHJcblxyXG4gIGFzeW5jIHBsYXlCdWZmZXJzKCkge1xyXG4gICAgbGV0IGF1ZGlvQmxvYiA9IGF3YWl0IHRoaXMuQnVmZmVyc1RvQmxvYih0aGlzLmFsbFJlY29yZGVkQnVmZmVycyk7XHJcbiAgICB0aGlzLnBsYXlQQ00oYXVkaW9CbG9iLCA0NDEwMCk7XHJcbiAgfVxyXG5cclxuICBCdWZmZXJzVG9CbG9iKGJ1ZmZlcnM6IEFycmF5PGFueT4pIHtcclxuICAgIGxldCBhdWRpb0J1ZmZlcjogYW55ID0gW107XHJcbiAgICBidWZmZXJzLmZvckVhY2goYnVmZmVyID0+IHtcclxuICAgICAgYnVmZmVyLmZvckVhY2goaW50MTYgPT4ge1xyXG4gICAgICAgIGF1ZGlvQnVmZmVyLnB1c2goaW50MTYpO1xyXG4gICAgICB9KTtcclxuICAgIH0pO1xyXG4gICAgcmV0dXJuIG5ldyBCbG9iKFthdWRpb0J1ZmZlcl0sIHsgdHlwZTogJ2F1ZGlvL3BjbScgfSk7XHJcbiAgfVxyXG59Il19
1
+
2
+ /**
3
+ * @copyright © 未来飞马 © 未来全栈 www.fmode.cn
4
+ * 版权所有 © 未来飞马 © 江西脑控科技有限公司 Copyright © Fmode Technology Co., Ltd.
5
+ * 保留所有权利 All Rights Reserved.
6
+ * /home/ryan/workspace/nova/nova-admin/dist/fmode-ng/esm2022/lib/aigc/voice/fmode-voice.service.mjs
7
+ */
8
+ import Recorder from"recorder-core";function checkWorkletSupport(){try{return window.AudioWorkletNode&&window.AudioContext&&window.AudioContext.audioWorklet&&!/Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent)}catch(e){return!1}}Recorder.ConnectEnableWorklet=checkWorkletSupport();import"recorder-core/src/engine/pcm";import"recorder-core/src/engine/wav";import"recorder-core/src/extensions/waveview";import CryptoJS from"crypto-js";import{pcmtoWav}from"./lib/pcm2wav";import{convertFrameBufferToBase64,resampleBuffer}from"./lib/resample";export class FmodeVoiceService{constructor(e,t){this.platform=e,this.diagnostic=t,this.disableASR=!1,this.isRecording=!1,this.isUserFinish=!1,this.shouldReconnect=!0,this.reconnectAttempts=0,this.maxReconnectAttempts=3,this.reconnectDelay=1e3,this.recordWavBlob=null,this.recordPcmBlob=null,this.recordDuration=0,this.allRecordedBuffers=[],this.currentRecorder=null,this.btnStatus="UNDEFINED",this.connStatus="",this.resultText="",this.resultTextTemp="",this.durationStr="00:00",this.duration=0,this.recordType="pcm",this.encodingType="raw",this.APPID="50f4a46c",this.API_SECRET="NzFlNmFhZDJjMDNkZGM3NzI0Mzg2OGNm",this.API_KEY="106ddc40dfd4b9ca6d7b47c70fada749",this.requestPermission()}toggleRecord(){console.log("toggleRecord",this.btnStatus),"UNDEFINED"===this.btnStatus||"CLOSED"===this.btnStatus?this.startTalk():"CONNECTING"!==this.btnStatus&&"OPEN"!==this.btnStatus||this.finishTalk()}async startTalk(e){console.log("startTalk called"),this.resetState(),this.onBeforeStartTalk&&this.onBeforeStartTalk(),e?.preventDefault();try{await this.openWithPriviledge(),await this.startIndependentRecording(),this.isRecording=!0,this.shouldReconnect=!0,this.reconnectAttempts=0,setTimeout((()=>{this.connectWebSocket()}),100),this.startCountdown(),this.onAfterStartTalk&&this.onAfterStartTalk()}catch(e){console.error("Failed to start talk:",e),this.resetState()}}async finishTalk(){console.log("finishTalk called"),this.isUserFinish=!0,this.shouldReconnect=!1,this.onBeforeFinishTalk&&this.onBeforeFinishTalk();try{this.closeWebSocket(),await this.stopIndependentRecording(),this.isRecording=!1,this.clearTimers(),this.changeBtnStatus("CLOSED"),setTimeout((()=>{this.isUserFinish&&(this.onAfterFinishTalk&&this.onAfterFinishTalk(),this.isUserFinish=!1)}),500)}catch(e){console.error("Error finishing talk:",e)}}cancelTalk(){console.log("cancelTalk called"),this.onBeforeCancelTalk&&this.onBeforeCancelTalk(),this.isUserFinish=!1,this.shouldReconnect=!1,this.isRecording=!1,this.closeWebSocket(),this.cancelIndependentRecording(),this.resetState(),this.onAfterCancelTalk&&this.onAfterCancelTalk()}async startIndependentRecording(){return console.log("Starting independent recording"),this.allRecordedBuffers=[],this.createIndependentRecorder(),new Promise(((e,t)=>{this.currentRecorder.open((()=>{console.log("Independent recorder opened"),document.querySelector(".record-wave")&&Recorder.WaveView&&(this.waveClient=Recorder.WaveView({elem:".record-wave"})),this.currentRecorder.start(),this.onAfterRecordStart&&this.onAfterRecordStart(),e()}),((e,o)=>{console.error("Failed to open independent recorder:",e),t(new Error(e))}))}))}async stopIndependentRecording(){if(console.log("Stopping independent recording"),this.currentRecorder)return new Promise((e=>{this.currentRecorder.stop((async(t,o)=>{console.log("Independent recording stopped successfully",t,o),this.recordPcmBlob=t,this.recordWavBlob=await this.pcmBlobToWavBlob(t,44100),this.cleanupIndependentRecorder(),e()}),(t=>{console.error("独立录音停止失败:"+t),this.cleanupIndependentRecorder(),e()}))}));console.log("No independent recorder to stop")}cancelIndependentRecording(){if(console.log("Cancelling independent recording"),this.currentRecorder)try{this.currentRecorder.close()}catch(e){console.error("Error closing independent recorder:",e)}this.cleanupIndependentRecorder(),this.recordPcmBlob=null,this.recordWavBlob=null}createIndependentRecorder(){this.currentRecorder&&this.cleanupIndependentRecorder(),this.currentRecorder=Recorder({type:this.recordType,sampleRate:44100,bitRate:16,onProcess:(e,t,o,r,i,s)=>{if(!this.isRecording)return;this.allRecordedBuffers=e;let n=e.length&&e[e.length-1];n&&(n=resampleBuffer(n,44100,16e3),this.sendAudioToWebSocket(n),this.waveClient?.input(e[e.length-1],t,r))}})}cleanupIndependentRecorder(){if(this.currentRecorder){try{this.currentRecorder.close()}catch(e){console.error("Error cleaning up independent recorder:",e)}this.currentRecorder=null}}sendAudioToWebSocket(e){if(this.iatWS&&this.iatWS.readyState===WebSocket.OPEN&&!this.disableASR)try{this.iatWS.send(JSON.stringify({data:{status:1,format:"audio/L16;rate=16000",encoding:this.encodingType,audio:convertFrameBufferToBase64(e)}}))}catch(e){console.error("Error sending audio data to WebSocket:",e)}}connectWebSocket(){if(console.log("connectWebSocket called, shouldReconnect:",this.shouldReconnect),!this.shouldReconnect||!this.isRecording)return void console.log("Not connecting WebSocket - shouldReconnect:",this.shouldReconnect,"isRecording:",this.isRecording);this.closeWebSocket();const e=this.getWebSocketUrl();console.log("Connecting to:",e);try{if(!("WebSocket"in window))return console.error("浏览器不支持WebSocket"),void alert("浏览器不支持WebSocket");this.iatWS=new WebSocket(e),this.changeBtnStatus("CONNECTING"),this.iatWS.onopen=e=>{console.log("WebSocket connected for ASR"),this.reconnectAttempts=0,this.changeBtnStatus("OPEN");const t={common:{app_id:this.APPID},business:{language:"zh_cn",domain:"iat",accent:"mandarin",vad_eos:1e4,dwa:"wpgs"},data:{status:0,format:"audio/L16;rate=16000",encoding:this.encodingType}};this.iatWS.send(JSON.stringify(t))},this.iatWS.onmessage=e=>{this.renderResult(e.data)},this.iatWS.onerror=e=>{console.error("WebSocket error:",e),this.handleWebSocketError()},this.iatWS.onclose=e=>{console.log("WebSocket closed:",e.code,e.reason),this.handleWebSocketClose(e)}}catch(e){console.error("Failed to create WebSocket:",e),this.handleWebSocketError()}}handleWebSocketClose(e){if(console.log("handleWebSocketClose",{code:e.code,reason:e.reason,shouldReconnect:this.shouldReconnect,isRecording:this.isRecording,isUserFinish:this.isUserFinish}),!this.isUserFinish&&this.shouldReconnect&&this.isRecording){try{this.iatWS&&this.iatWS.readyState===WebSocket.OPEN&&this.iatWS.send(JSON.stringify({data:{status:2,format:"audio/L16;rate=16000",encoding:this.encodingType,audio:""}}))}catch(e){console.error("Error sending end frame on close:",e)}1e3===e.code||1006===e.code||1011===e.code?this.attemptReconnect():(console.error("WebSocket closed with error code:",e.code),this.handleWebSocketError())}}handleWebSocketError(){this.shouldReconnect&&this.isRecording&&this.attemptReconnect()}attemptReconnect(){this.reconnectAttempts>=this.maxReconnectAttempts?console.error("Max reconnect attempts reached"):(this.reconnectAttempts++,console.log(`Attempting WebSocket reconnect ${this.reconnectAttempts}/${this.maxReconnectAttempts}`),setTimeout((()=>{this.shouldReconnect&&this.isRecording&&!this.isUserFinish&&this.connectWebSocket()}),this.reconnectDelay*this.reconnectAttempts))}closeWebSocket(){if(this.iatWS){try{this.iatWS.readyState===WebSocket.OPEN&&this.iatWS.send(JSON.stringify({data:{status:2,format:"audio/L16;rate=16000",encoding:this.encodingType,audio:""}})),this.iatWS.readyState!==WebSocket.OPEN&&this.iatWS.readyState!==WebSocket.CONNECTING||this.iatWS.close(1e3,"User initiated close")}catch(e){console.error("Error closing WebSocket:",e)}this.iatWS=null}}resetState(){this.resultText="",this.resultTextTemp="",this.durationStr="00:00",this.duration=0,this.recordDuration=0,this.allRecordedBuffers=[],this.clearTimers()}clearTimers(){this.countdownInterval&&(clearInterval(this.countdownInterval),this.countdownInterval=null),this.durationInterval&&(clearInterval(this.durationInterval),this.durationInterval=null)}changeBtnStatus(e){switch(this.btnStatus=e,e){case"CONNECTING":this.connStatus="建立连接中";break;case"OPEN":this.connStatus=`录音中(${this.durationStr})`;break;case"CLOSING":this.connStatus="关闭连接中";break;case"CLOSED":this.connStatus="开始录音"}}startCountdown(){this.clearTimers(),this.recordDuration=0,this.duration=0,this.now=new Date,this.durationInterval=setInterval((()=>{this.isRecording&&(this.recordDuration+=100)}),100),this.countdownInterval=setInterval((()=>{this.isRecording&&this.countTimer()}),1e3)}countTimer(){this.duration++;let e=String(Math.floor(this.duration/60)).padStart(2,"0")+":"+String(this.duration%60).padStart(2,"0");this.durationStr=e,this.connStatus=`录音中(${this.durationStr})`,this.onDurationStrChange&&this.onDurationStrChange(e)}renderResult(e){try{let t=JSON.parse(e);if(t.data&&t.data.result){let e=t.data.result,o="",r=e.ws;for(let e=0;e<r.length;e++)o+=r[e].cw[0].w;e.pgs?("apd"===e.pgs&&(this.resultText=this.resultTextTemp),this.resultTextTemp=this.resultText+o):this.resultText=this.resultText+o,this.onInputChange&&this.onInputChange(this.getUserInput())}0===t.code&&2===t.data.status&&(console.log("ASR session completed, will reconnect if still recording"),this.iatWS&&this.iatWS.readyState===WebSocket.OPEN&&this.iatWS.close()),0!==t.code&&(console.error("ASR error:",t),this.iatWS&&this.iatWS.readyState===WebSocket.OPEN&&this.iatWS.close())}catch(e){console.error("Error parsing result:",e)}}getUserInput(){return""+(this.resultTextTemp||this.resultText)}async openWithPriviledge(){return await this.requestPermission(),Recorder.IsOpen(),!0}async pcmBlobToWavBlob(e,t){return new Promise((o=>{let r=new FileReader;r.onload=function(e){let r=e.target.result,i=pcmtoWav(r,t,1,16);o(i)},r.readAsArrayBuffer(e)}))}async playPCM(e,t){let o=await this.pcmBlobToWavBlob(e,t),r=window.URL.createObjectURL(o),i=new Audio;i.src=r,i.play()}playRecord(){this.recordPcmBlob&&this.playPCM(this.recordPcmBlob,44100)}getWebSocketUrl(){let e="wss://iat-api.xfyun.cn/v2/iat",t="iat-api.xfyun.cn",o=this.API_KEY,r=this.API_SECRET,i=(new Date).toUTCString(),s=`host: ${t}\ndate: ${i}\nGET /v2/iat HTTP/1.1`,n=CryptoJS.HmacSHA256(s,r),c=CryptoJS.enc.Base64.stringify(n);return e=`${e}?authorization=${btoa(`api_key="${o}", algorithm="hmac-sha256", headers="host date request-line", signature="${c}"`)}&date=${i}&host=${t}`,e}toBase64(e){for(var t="",o=new Uint8Array(e),r=o.byteLength,i=0;i<r;i++)t+=String.fromCharCode(o[i]);return window.btoa(t)}isCapacitor(){return!!this.platform?.is&&(this.platform.is("capacitor")||this.platform.is("cordova"))}async requestPermission(){if(this.isCapacitor())try{await this.requestStoagePermission(),await this.requestCameraPermission(),await this.requestMicPermission(),await this.requestRecordAudioPermission()}catch(e){console.error(e)}}async requestRecordAudioPermission(){let e=await this.diagnostic.requestRuntimePermissions([this.diagnostic.permission.RECORD_AUDIO]);console.log("record permission request:",e)}async requestMicPermission(){let e=await this.diagnostic.isMicrophoneAuthorized();if(console.log("permisson_MIC:",e),!e){await this.diagnostic.requestMicrophoneAuthorization()}}async requestStoagePermission(){let e=await this.diagnostic.isExternalStorageAuthorized();if(console.log("permisson_STORAGE:",e),!e){await this.diagnostic.requestExternalStorageAuthorization()}}async requestCameraPermission(){let e=await this.diagnostic.isCameraAuthorized();if(console.log("permisson_Camera:",e),!e){await this.diagnostic.requestCameraAuthorization()}}splitAudioData(e){const t=1280,o=Math.ceil(e.length/t),r=[];for(let i=0;i<o;i++){const o=i*t,s=o+t,n=e.slice(o,s);r.push(n)}return r}BufferToBlob(e){return new Blob([e],{type:"audio/pcm"})}async playBuffers(){let e=await this.BuffersToBlob(this.allRecordedBuffers);this.playPCM(e,44100)}BuffersToBlob(e){let t=[];return e.forEach((e=>{e.forEach((e=>{t.push(e)}))})),new Blob([t],{type:"audio/pcm"})}}
9
+ var MODULE_PATH_NEED = `6K+l5paH5Lu25piv5pys6aG555uu55qE5LiA6YOo5YiGIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBDb21wb25lbnRzIGluIEZtb2RlIEluYy4KICAgIOeJiOadg+aJgOaciSDCqSDmnKrmnaXpo57pqawgwqkg5rGf6KW/6ISR5o6n56eR5oqA5pyJ6ZmQ5YWs5Y+4IENvcHlyaWdodCDCqSBGbW9kZSBUZWNobm9sb2d5IENvLiwgTHRkLgogICAg5L+d55WZ5omA5pyJ5p2D5YipIEFsbCBSaWdodHMgUmVzZXJ2ZWQuCiAgICDkuKXnpoHlnKjmnKrnu4/mjojmnYPnmoTmg4XlhrXkuIvvvIzpgJrov4fku7vkvZXlqpLku4vlpI3liLbmraTmlofku7YgVW5hdXRob3JpemVkIGNvcHlpbmcgb2YgdGhpcyBmaWxlLCB2aWEgYW55IG1lZGl1bSBpcyBzdHJpY3RseSBwcm9oaWJpdGVkCiAgICDor6Xmlofku7bmmK/kuJPmnInnmoTmnLrlr4bmlofku7YgUHJvcHJpZXRhcnkgYW5kIGNvbmZpZGVudGlhbAogICAKICAgIENvcHlyaWdodCAyMDIxLW5vdyBGbW9kZSBJbmMuIHN1cHBvcnRAZm1vZGUuY24uIDE4NjA3MDA3MDczLgogICAg5L+d55WZ5omA5pyJ5p2D5YipIEFsbCByaWdodHMgcmVzZXJ2ZWQuCgogICAgUEFUSDovaG9tZS9yeWFuL3dvcmtzcGFjZS9ub3ZhL25vdmEtYWRtaW4vZGlzdC9mbW9kZS1uZy9lc20yMDIyL2xpYi9haWdjL3ZvaWNlL2Ztb2RlLXZvaWNlLnNlcnZpY2UubWpz`
10
+