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.
- package/LICENSE.md +8 -0
- package/README.md +37 -37
- package/esm2022/fmode-ng.mjs +10 -5
- package/esm2022/lib/aigc/agent/fm-agent-task/fm-agent-task.component.mjs +8 -92
- package/esm2022/lib/aigc/agent/index.mjs +10 -3
- package/esm2022/lib/aigc/avatar/avatar.module.mjs +10 -45
- package/esm2022/lib/aigc/avatar/comp-avatar-particle/avatar.role.mjs +10 -2
- package/esm2022/lib/aigc/avatar/comp-avatar-particle/comp-avatar-particle.component.mjs +10 -315
- package/esm2022/lib/aigc/avatar/comp-avatar-particle/index.mjs +10 -3
- package/esm2022/lib/aigc/avatar/comp-avatar-particle/role-points.class.mjs +10 -57
- package/esm2022/lib/aigc/avatar/comp-avatar-role-image/comp-avatar-role-image.component.mjs +10 -98
- package/esm2022/lib/aigc/avatar/comp-avatar-role-video/comp-avatar-role-video.component.mjs +10 -107
- package/esm2022/lib/aigc/avatar/comp-avatar-talk/comp-avatar-talk.component.mjs +10 -113
- package/esm2022/lib/aigc/avatar/index.mjs +10 -8
- package/esm2022/lib/aigc/avatar/interface-avatar-role.mjs +10 -2
- package/esm2022/lib/aigc/avatar/modal-chat-voice-input/modal-chat-voice-input.component.mjs +8 -164
- package/esm2022/lib/aigc/chat/chat-header-area/comp-header-area.component.mjs +10 -49
- package/esm2022/lib/aigc/chat/chat-header-area/index.mjs +10 -2
- package/esm2022/lib/aigc/chat/chat-list/chat-list.component.mjs +8 -155
- package/esm2022/lib/aigc/chat/chat-list/index.mjs +10 -2
- package/esm2022/lib/aigc/chat/chat-message-area/comp-message-area.component.mjs +10 -40
- package/esm2022/lib/aigc/chat/chat-message-area/index.mjs +10 -2
- package/esm2022/lib/aigc/chat/chat-message-card/comp-message-card.component.mjs +10 -140
- package/esm2022/lib/aigc/chat/chat-message-card/duration-str.pipe.mjs +10 -29
- package/esm2022/lib/aigc/chat/chat-message-card/index.mjs +10 -3
- package/esm2022/lib/aigc/chat/chat-modal-input/index.mjs +10 -3
- package/esm2022/lib/aigc/chat/chat-modal-input/modal-audio-message/modal-audio-message.component.mjs +8 -193
- package/esm2022/lib/aigc/chat/chat-modal-input/modal-input.component.mjs +10 -331
- package/esm2022/lib/aigc/chat/chat-panel/chat-panel.component.mjs +8 -249
- package/esm2022/lib/aigc/chat/comp-role-prompt/comp-role-prompt.component.mjs +10 -83
- package/esm2022/lib/aigc/chat/comp-role-prompt/index.mjs +10 -2
- package/esm2022/lib/aigc/chat/index.mjs +10 -8
- package/esm2022/lib/aigc/comp-markdown-preview/clipboard.service.mjs +10 -82
- package/esm2022/lib/aigc/comp-markdown-preview/markdown-parse.mjs +8 -367
- package/esm2022/lib/aigc/comp-markdown-preview/markdown-preview.component.mjs +10 -51
- package/esm2022/lib/aigc/comp-markdown-preview/markdown-preview.module.mjs +10 -24
- package/esm2022/lib/aigc/comp-markdown-preview/plugins/md-mathjax/index.mjs +10 -115
- package/esm2022/lib/aigc/index.mjs +10 -14
- package/esm2022/lib/aigc/service-fmai/fmai.service.mjs +10 -21
- package/esm2022/lib/aigc/service-fmai/service-chat/chat-class.mjs +10 -2
- package/esm2022/lib/aigc/service-fmai/service-chat/chat.service.mjs +8 -174
- package/esm2022/lib/aigc/service-fmai/service-chat/index.mjs +10 -7
- package/esm2022/lib/aigc/service-fmai/service-chat/mask-list.mjs +9 -194
- package/esm2022/lib/aigc/service-fmai/service-chat/pipes/chat-content.pipe.mjs +10 -27
- package/esm2022/lib/aigc/service-fmai/service-chat/pipes/hidexml.pipe.mjs +10 -27
- package/esm2022/lib/aigc/service-fmai/service-chat/utilnow.pipe.mjs +10 -68
- package/esm2022/lib/aigc/service-fmai/service-imagine/imagine-func.mjs +9 -162
- package/esm2022/lib/aigc/service-fmai/service-imagine/imagine-work.mjs +10 -68
- package/esm2022/lib/aigc/service-fmai/service-imagine/imagine.service.mjs +8 -313
- package/esm2022/lib/aigc/service-fmai/service-imagine/index.mjs +10 -4
- package/esm2022/lib/aigc/story/fm-office-viewer/fm-office-viewer.component.mjs +10 -62
- package/esm2022/lib/aigc/story/fm-story-card/fm-story-card.component.mjs +10 -87
- package/esm2022/lib/aigc/story/fm-story-list/fm-story-list.component.mjs +10 -345
- package/esm2022/lib/aigc/story/fm-story-list/story-preview.mjs +10 -81
- package/esm2022/lib/aigc/story/fm-story-loader/fm-story-loader.component.mjs +10 -152
- package/esm2022/lib/aigc/story/fm-story-splitter/fm-story-splitter.component.mjs +10 -42
- package/esm2022/lib/aigc/story/index.mjs +10 -6
- package/esm2022/lib/aigc/story/modal-chat-story/comp-chat-story-json/comp-chat-story-json.component.mjs +10 -47
- package/esm2022/lib/aigc/story/modal-chat-story/comp-diary-story/comp-diary-story.component.mjs +10 -95
- package/esm2022/lib/aigc/story/modal-chat-story/modal-chat-story.component.mjs +10 -253
- package/esm2022/lib/aigc/story/story.service.mjs +10 -35
- package/esm2022/lib/aigc/voice/fmode-voice.service.mjs +10 -636
- package/esm2022/lib/aigc/voice/index.mjs +10 -7
- package/esm2022/lib/aigc/voice/lib/audio/audio.player.mjs +10 -55
- package/esm2022/lib/aigc/voice/lib/audio/audio.streamer.mjs +10 -2
- package/esm2022/lib/aigc/voice/lib/audio/streamer.microsoft.mjs +10 -96
- package/esm2022/lib/aigc/voice/lib/audio/streamer.pcm.mjs +8 -177
- package/esm2022/lib/aigc/voice/lib/pcm2wav.mjs +10 -38
- package/esm2022/lib/aigc/voice/lib/recorder/extension-waveview.mjs +10 -215
- package/esm2022/lib/aigc/voice/lib/resample.mjs +10 -34
- package/esm2022/lib/aigc/voice/tts/fmode-tts-class.mjs +10 -189
- package/esm2022/lib/aigc/voice/tts/index.mjs +10 -5
- package/esm2022/lib/aigc/voice/tts/int-tts-provider.mjs +10 -2
- package/esm2022/lib/aigc/voice/tts/provider-doubao.mjs +8 -346
- package/esm2022/lib/code/fm-codemirror/fm-codemirror.component.mjs +8 -342
- package/esm2022/lib/code/index.mjs +10 -2
- package/esm2022/lib/core/agent/chat/completion/fmode-completion.mjs +8 -430
- package/esm2022/lib/core/agent/chat/completion/index.mjs +10 -2
- package/esm2022/lib/core/agent/chat/completion/int-gpt-chat-options.mjs +10 -2
- package/esm2022/lib/core/agent/chat/fmode-chat.mjs +8 -617
- package/esm2022/lib/core/agent/chat/index.mjs +10 -4
- package/esm2022/lib/core/agent/chat/interface.mjs +10 -2
- package/esm2022/lib/core/agent/index.mjs +10 -6
- package/esm2022/lib/core/agent/prompt/agent.prompt.mjs +10 -133
- package/esm2022/lib/core/agent/prompt/index.mjs +10 -2
- package/esm2022/lib/core/agent/prompt/prompt-util.mjs +10 -16
- package/esm2022/lib/core/agent/story/agent.story.mjs +10 -64
- package/esm2022/lib/core/agent/story/index.mjs +10 -2
- package/esm2022/lib/core/agent/task/agent.task.mjs +10 -90
- package/esm2022/lib/core/agent/task/index.mjs +10 -2
- package/esm2022/lib/core/agent/waiting/index.mjs +10 -3
- package/esm2022/lib/core/agent/waiting/loading/loading.ctrl.mjs +10 -279
- package/esm2022/lib/core/agent/waiting/tips/tips.ctrl.mjs +8 -190
- package/esm2022/lib/core/index.mjs +10 -4
- package/esm2022/lib/core/parse/datatype/acl.mjs +10 -57
- package/esm2022/lib/core/parse/datatype/file.mjs +10 -69
- package/esm2022/lib/core/parse/datatype/geopoint.mjs +10 -57
- package/esm2022/lib/core/parse/datatype/relation.mjs +10 -56
- package/esm2022/lib/core/parse/fmode.cloud.mjs +10 -0
- package/esm2022/lib/core/parse/fmode.object.mjs +10 -232
- package/esm2022/lib/core/parse/fmode.parse.mjs +10 -84
- package/esm2022/lib/core/parse/fmode.query.mjs +8 -500
- package/esm2022/lib/core/parse/fmode.user.mjs +10 -275
- package/esm2022/lib/core/parse/index.mjs +10 -5
- package/esm2022/lib/core/parse/types.mjs +10 -2
- package/esm2022/lib/core/voice/index.mjs +10 -3
- package/esm2022/lib/core/voice/tts/index.mjs +10 -2
- package/esm2022/lib/icon/filetype/audio.svg.mjs +10 -3
- package/esm2022/lib/icon/filetype/avatar.svg.mjs +10 -14
- package/esm2022/lib/icon/filetype/chat.svg.mjs +10 -3
- package/esm2022/lib/icon/filetype/docx.svg.mjs +10 -3
- package/esm2022/lib/icon/filetype/file.svg.mjs +10 -3
- package/esm2022/lib/icon/filetype/filetype.pipe.mjs +10 -57
- package/esm2022/lib/icon/filetype/index.mjs +10 -12
- package/esm2022/lib/icon/filetype/md.svg.mjs +10 -3
- package/esm2022/lib/icon/filetype/pdf.svg.mjs +10 -3
- package/esm2022/lib/icon/filetype/pptx.svg.mjs +10 -3
- package/esm2022/lib/icon/filetype/svgtoblob.mjs +10 -8
- package/esm2022/lib/icon/filetype/video.svg.mjs +10 -3
- package/esm2022/lib/icon/filetype/xlsx.svg.mjs +10 -3
- package/esm2022/lib/icon/index.mjs +10 -2
- package/esm2022/lib/map/comp-poi-picker/comp-poi-picker.component.mjs +10 -209
- package/esm2022/lib/map/comp-poi-picker/comp-poi-picker.module.mjs +10 -33
- package/esm2022/lib/map/index.mjs +10 -4
- package/esm2022/lib/map/map.module.mjs +10 -61
- package/esm2022/lib/map/page-loca-scatter/page-loca-scatter.component.mjs +10 -132
- package/esm2022/lib/map/page-map.start/page-map.start.component.mjs +8 -115
- package/esm2022/lib/map/page-plan-route/page-plan-route.component.mjs +8 -118
- package/esm2022/lib/nova-cloud/index.mjs +10 -3
- package/esm2022/lib/nova-cloud/ncloud-api-func.mjs +10 -91
- package/esm2022/lib/nova-cloud/nova-cloud.service.mjs +10 -53
- package/esm2022/lib/payment/index.mjs +10 -2
- package/esm2022/lib/payment/payment/payment.component.mjs +10 -543
- package/esm2022/lib/payment/payment.service.mjs +10 -202
- package/esm2022/lib/person/comp-person-gender-icon/comp-person-gender-icon.component.mjs +10 -48
- package/esm2022/lib/person/comp-person-item/comp-person-item.component.mjs +10 -29
- package/esm2022/lib/person/comp-person-story/comp-person-story.component.mjs +10 -211
- package/esm2022/lib/person/edit-upload/edit-upload.component.mjs +8 -412
- package/esm2022/lib/person/edit-upload/edit-upload.module.mjs +10 -50
- package/esm2022/lib/person/index.mjs +10 -5
- package/esm2022/lib/person/modal-person-select/modal-person-select.component.mjs +10 -144
- package/esm2022/lib/person/modal-user-verify/secret-text.pipe.mjs +10 -41
- package/esm2022/lib/person/modal-user-verify/user-verify.component.mjs +10 -595
- package/esm2022/lib/person/person-detail/person-detail.component.mjs +10 -172
- package/esm2022/lib/person/person.service.mjs +8 -193
- package/esm2022/lib/platform/cross.service.mjs +10 -62
- package/esm2022/lib/platform/index.mjs +10 -2
- package/esm2022/lib/social/index.mjs +10 -2
- package/esm2022/lib/social/wechat/wechat-jssdk.service.mjs +8 -230
- package/esm2022/lib/storage/comp-hwobs-manager/hwobs-manager.component.mjs +10 -59
- package/esm2022/lib/storage/index.mjs +10 -5
- package/esm2022/lib/storage/service-hwobs/hwobs.service.mjs +8 -131
- package/esm2022/lib/storage/service-hwobs/index.mjs +10 -3
- package/esm2022/lib/storage/service-hwobs/typings/esdk-obs-browser.mjs +1 -1
- package/esm2022/lib/storage/service-upload/index.mjs +10 -2
- package/esm2022/lib/storage/service-upload/nova-upload.service.mjs +8 -513
- package/esm2022/lib/storage/service-upload/util-file-md5.mjs +10 -28
- package/esm2022/lib/storage/service-upload/util-file-metadata.mjs +10 -93
- package/esm2022/lib/storage/storage.module.mjs +10 -42
- package/esm2022/lib/text/fm-article-editor/article-editor-topbar/article-editor-topbar.component.mjs +10 -72
- package/esm2022/lib/text/fm-article-editor/article.service.mjs +10 -237
- package/esm2022/lib/text/fm-article-editor/comp-upload-book-banners/comp-upload-book-banners.component.mjs +8 -72
- package/esm2022/lib/text/fm-article-editor/draft.service.mjs +10 -207
- package/esm2022/lib/text/fm-article-editor/fm-article-aitool/fm-article-aitool.component.mjs +8 -227
- package/esm2022/lib/text/fm-article-editor/fm-article-draft/fm-article-draft.component.mjs +10 -303
- package/esm2022/lib/text/fm-article-editor/fm-article-editor.component.mjs +8 -371
- package/esm2022/lib/text/fm-article-editor/fm-article-outline/fm-article-outline.component.mjs +10 -94
- package/esm2022/lib/text/fm-article-editor/fm-article-outline-leftitem/fm-article-outline-leftitem.component.mjs +10 -36
- package/esm2022/lib/text/fm-article-editor/fm-article-preview/fm-article-preview.component.mjs +10 -281
- package/esm2022/lib/text/fm-article-editor/fm-article-write-options/fm-article-write-options.component.mjs +10 -72
- package/esm2022/lib/text/fm-article-editor/outline-count.pipe.mjs +10 -22
- package/esm2022/lib/text/fm-article-editor/prompt/prompt-insertion-article.mjs +10 -169
- package/esm2022/lib/text/fm-article-editor/task-article-generation.mjs +10 -69
- package/esm2022/lib/text/fm-article-editor/tasks/task-article-draft-create.mjs +10 -65
- package/esm2022/lib/text/fm-article-editor/tasks/task-article-outline-edit.mjs +10 -55
- package/esm2022/lib/text/fm-article-editor/tasks/task-article-outline.mjs +10 -142
- package/esm2022/lib/text/fm-article-editor/tasks/task-article-preview.mjs +10 -35
- package/esm2022/lib/text/fm-article-editor/tasks/task-article-writing-options.mjs +10 -81
- package/esm2022/lib/text/fm-article-editor/tasks/task-document-select.mjs +10 -50
- package/esm2022/lib/text/fm-text-quill/fm-text-quill.component.mjs +10 -145
- package/esm2022/lib/text/index.mjs +10 -4
- package/esm2022/lib/user/account/account.service.mjs +10 -222
- package/esm2022/lib/user/captcha/captcha.component.mjs +10 -135
- package/esm2022/lib/user/comp-user-avatar/comp-user-avatar.component.mjs +10 -66
- package/esm2022/lib/user/index.mjs +10 -17
- package/esm2022/lib/user/login/auth.guard.mjs +10 -28
- package/esm2022/lib/user/login/auth.service.mjs +8 -433
- package/esm2022/lib/user/login/login.component.mjs +10 -916
- package/esm2022/lib/user/modal-user-login/modal-user-login.component.mjs +10 -311
- package/esm2022/lib/user/profile/auth-profile.guard.mjs +10 -27
- package/esm2022/lib/user/profile/auth-profile.service.mjs +10 -122
- package/esm2022/lib/user/profile/profile-bind/profile-bind.component.mjs +10 -164
- package/esm2022/lib/user/profile/profile-bind/profile-confirm-modal.component.mjs +10 -79
- package/esm2022/lib/user/profile/profile.module.mjs +10 -54
- package/esm2022/lib/user/staff/index.mjs +10 -4
- package/esm2022/lib/user/staff/staff.guard.mjs +10 -26
- package/esm2022/lib/user/staff/staff.module.mjs +10 -18
- package/esm2022/lib/user/staff/staff.service.mjs +10 -85
- package/esm2022/lib/user/user-name.pipe.mjs +10 -29
- package/esm2022/lib/user/user.module.mjs +10 -102
- package/esm2022/lib/video/fm-video/fm-video.component.mjs +10 -67
- package/esm2022/lib/video/index.mjs +10 -2
- package/esm2022/public-api.mjs +10 -21
- package/fesm2022/fmode-ng.mjs +7 -19788
- package/fesm2022/fmode-ng.mjs.map +1 -1
- package/lib/aigc/avatar/comp-avatar-role-image/comp-avatar-role-image.component.d.ts +2 -2
- package/lib/aigc/avatar/comp-avatar-role-video/comp-avatar-role-video.component.d.ts +2 -2
- package/lib/aigc/avatar/comp-avatar-talk/comp-avatar-talk.component.d.ts +2 -2
- package/lib/aigc/chat/chat-list/chat-list.component.d.ts +3 -2
- package/lib/aigc/chat/chat-message-card/comp-message-card.component.d.ts +3 -3
- package/lib/aigc/chat/chat-modal-input/modal-audio-message/modal-audio-message.component.d.ts +2 -2
- package/lib/aigc/chat/chat-modal-input/modal-input.component.d.ts +3 -3
- package/lib/aigc/service-fmai/service-imagine/imagine-func.d.ts +3 -3
- package/lib/aigc/service-fmai/service-imagine/imagine-work.d.ts +4 -4
- package/lib/aigc/service-fmai/service-imagine/imagine.service.d.ts +6 -6
- package/lib/aigc/story/fm-office-viewer/fm-office-viewer.component.d.ts +2 -2
- package/lib/aigc/story/fm-story-card/fm-story-card.component.d.ts +2 -1
- package/lib/aigc/story/fm-story-list/fm-story-list.component.d.ts +6 -6
- package/lib/aigc/story/fm-story-list/story-preview.d.ts +4 -4
- package/lib/aigc/story/fm-story-loader/fm-story-loader.component.d.ts +3 -3
- package/lib/aigc/story/fm-story-splitter/fm-story-splitter.component.d.ts +2 -2
- package/lib/aigc/story/modal-chat-story/comp-chat-story-json/comp-chat-story-json.component.d.ts +3 -2
- package/lib/aigc/story/modal-chat-story/comp-diary-story/comp-diary-story.component.d.ts +3 -2
- package/lib/aigc/story/modal-chat-story/modal-chat-story.component.d.ts +6 -6
- package/lib/aigc/story/story.service.d.ts +3 -2
- package/lib/aigc/voice/tts/fmode-tts-class.d.ts +2 -2
- package/lib/core/agent/chat/fmode-chat.d.ts +11 -11
- package/lib/core/agent/story/agent.story.d.ts +8 -8
- package/lib/core/parse/datatype/geopoint.d.ts +5 -1
- package/lib/core/parse/fmode.cloud.d.ts +32 -0
- package/lib/core/parse/fmode.object.d.ts +2 -2
- package/lib/core/parse/fmode.parse.d.ts +15 -9
- package/lib/core/parse/fmode.query.d.ts +41 -2
- package/lib/core/parse/fmode.user.d.ts +1 -0
- package/lib/core/parse/index.d.ts +1 -0
- package/lib/core/parse/types.d.ts +2 -0
- package/lib/map/comp-poi-picker/comp-poi-picker.component.d.ts +5 -5
- package/lib/payment/payment/payment.component.d.ts +3 -3
- package/lib/payment/payment.service.d.ts +4 -4
- package/lib/person/comp-person-gender-icon/comp-person-gender-icon.component.d.ts +2 -1
- package/lib/person/comp-person-item/comp-person-item.component.d.ts +2 -1
- package/lib/person/comp-person-story/comp-person-story.component.d.ts +13 -13
- package/lib/person/edit-upload/edit-upload.component.d.ts +1 -1
- package/lib/person/modal-person-select/modal-person-select.component.d.ts +5 -5
- package/lib/person/modal-user-verify/user-verify.component.d.ts +5 -5
- package/lib/person/person-detail/person-detail.component.d.ts +4 -4
- package/lib/person/person.service.d.ts +7 -7
- package/lib/storage/service-hwobs/hwobs.service.d.ts +4 -5
- package/lib/storage/service-upload/nova-upload.service.d.ts +5 -5
- package/lib/text/fm-article-editor/article-editor-topbar/article-editor-topbar.component.d.ts +2 -1
- package/lib/text/fm-article-editor/article.service.d.ts +12 -12
- package/lib/text/fm-article-editor/draft.service.d.ts +5 -5
- package/lib/text/fm-article-editor/fm-article-aitool/fm-article-aitool.component.d.ts +3 -2
- package/lib/text/fm-article-editor/fm-article-draft/fm-article-draft.component.d.ts +16 -17
- package/lib/text/fm-article-editor/fm-article-editor.component.d.ts +3 -3
- package/lib/text/fm-article-editor/fm-article-outline/fm-article-outline.component.d.ts +11 -10
- package/lib/text/fm-article-editor/fm-article-preview/fm-article-preview.component.d.ts +2 -2
- package/lib/text/fm-article-editor/prompt/prompt-insertion-article.d.ts +3 -3
- package/lib/text/fm-article-editor/task-article-generation.d.ts +5 -4
- package/lib/text/fm-article-editor/tasks/task-article-draft-create.d.ts +3 -3
- package/lib/text/fm-article-editor/tasks/task-article-outline-edit.d.ts +3 -3
- package/lib/text/fm-article-editor/tasks/task-article-outline.d.ts +3 -3
- package/lib/text/fm-article-editor/tasks/task-article-preview.d.ts +2 -1
- package/lib/text/fm-article-editor/tasks/task-article-writing-options.d.ts +3 -3
- package/lib/text/fm-article-editor/tasks/task-document-select.d.ts +3 -3
- package/lib/user/account/account.service.d.ts +2 -2
- package/lib/user/comp-user-avatar/comp-user-avatar.component.d.ts +2 -2
- package/lib/user/login/auth.service.d.ts +5 -5
- package/lib/user/login/login.component.d.ts +3 -3
- package/lib/user/profile/auth-profile.service.d.ts +8 -8
- package/lib/user/profile/profile-bind/profile-bind.component.d.ts +7 -7
- package/lib/user/profile/profile-bind/profile-confirm-modal.component.d.ts +2 -2
- package/lib/user/staff/staff.service.d.ts +3 -3
- package/lib/user/user-name.pipe.d.ts +2 -2
- package/package.json +12 -18
|
@@ -1,636 +1,10 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
window.AudioContext &&
|
|
9
|
-
|
|
10
|
-
|
|
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
|
+
|