speechrecorderng 2.21.4 → 2.22.0

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 (111) hide show
  1. package/{esm2015/lib/action/action.js → esm2020/lib/action/action.mjs} +0 -0
  2. package/{esm2015/lib/audio/audio_display.js → esm2020/lib/audio/audio_display.mjs} +5 -5
  3. package/esm2020/lib/audio/audio_player.mjs +276 -0
  4. package/esm2020/lib/audio/capture/capture.mjs +547 -0
  5. package/{esm2015/lib/audio/context.js → esm2020/lib/audio/context.mjs} +0 -0
  6. package/esm2020/lib/audio/dsp/level_measure.mjs +351 -0
  7. package/{esm2015/lib/audio/format.js → esm2020/lib/audio/format.mjs} +0 -0
  8. package/{esm2015/lib/audio/impl/wavformat.js → esm2020/lib/audio/impl/wavformat.mjs} +0 -0
  9. package/{esm2015/lib/audio/impl/wavwriter.js → esm2020/lib/audio/impl/wavwriter.mjs} +0 -0
  10. package/{esm2015/lib/audio/io/stream.js → esm2020/lib/audio/io/stream.mjs} +0 -0
  11. package/{esm2015/lib/audio/persistor.js → esm2020/lib/audio/persistor.mjs} +0 -0
  12. package/{esm2015/lib/audio/playback/player.js → esm2020/lib/audio/playback/player.mjs} +0 -0
  13. package/esm2020/lib/audio/ui/audio_canvas_layer_comp.mjs +376 -0
  14. package/{esm2015/lib/audio/ui/audio_display_control.js → esm2020/lib/audio/ui/audio_display_control.mjs} +5 -5
  15. package/{esm2015/lib/audio/ui/audio_display_scroll_pane.js → esm2020/lib/audio/ui/audio_display_scroll_pane.mjs} +5 -5
  16. package/{esm2015/lib/audio/ui/audiosignal.js → esm2020/lib/audio/ui/audiosignal.mjs} +5 -5
  17. package/{esm2015/lib/audio/ui/common.js → esm2020/lib/audio/ui/common.mjs} +0 -0
  18. package/esm2020/lib/audio/ui/container.mjs +431 -0
  19. package/{esm2015/lib/audio/ui/livelevel.js → esm2020/lib/audio/ui/livelevel.mjs} +5 -5
  20. package/{esm2015/lib/audio/ui/scroll_pane_horizontal.js → esm2020/lib/audio/ui/scroll_pane_horizontal.mjs} +4 -4
  21. package/{esm2015/lib/audio/ui/sonagram.js → esm2020/lib/audio/ui/sonagram.mjs} +5 -5
  22. package/{esm2015/lib/dsp/utils.js → esm2020/lib/dsp/utils.mjs} +0 -0
  23. package/{esm2015/lib/io/BinaryWriter.js → esm2020/lib/io/BinaryWriter.mjs} +0 -0
  24. package/{esm2015/lib/io/stream.js → esm2020/lib/io/stream.mjs} +0 -0
  25. package/{esm2015/lib/math/2d/geometry.js → esm2020/lib/math/2d/geometry.mjs} +0 -0
  26. package/{esm2015/lib/math/complex.js → esm2020/lib/math/complex.mjs} +0 -0
  27. package/{esm2015/lib/math/dft.js → esm2020/lib/math/dft.mjs} +0 -0
  28. package/{esm2015/lib/math/utils.js → esm2020/lib/math/utils.mjs} +0 -0
  29. package/esm2020/lib/media/utils.mjs +14 -0
  30. package/esm2020/lib/net/uploader.mjs +246 -0
  31. package/{esm2015/lib/speechrecorder/project/project.js → esm2020/lib/speechrecorder/project/project.mjs} +0 -0
  32. package/{esm2015/lib/speechrecorder/project/project.service.js → esm2020/lib/speechrecorder/project/project.service.mjs} +4 -4
  33. package/esm2020/lib/speechrecorder/recording.mjs +54 -0
  34. package/esm2020/lib/speechrecorder/recordings/recordings.service.mjs +246 -0
  35. package/{esm2015/lib/speechrecorder/script/script.js → esm2020/lib/speechrecorder/script/script.mjs} +2 -2
  36. package/{esm2015/lib/speechrecorder/script/script.service.js → esm2020/lib/speechrecorder/script/script.service.mjs} +4 -4
  37. package/esm2020/lib/speechrecorder/session/audiorecorder.mjs +1144 -0
  38. package/esm2020/lib/speechrecorder/session/controlpanel.mjs +449 -0
  39. package/{esm2015/lib/speechrecorder/session/item.js → esm2020/lib/speechrecorder/session/item.mjs} +1 -1
  40. package/{esm2015/lib/speechrecorder/session/progress.js → esm2020/lib/speechrecorder/session/progress.mjs} +5 -5
  41. package/{esm2015/lib/speechrecorder/session/prompting.js → esm2020/lib/speechrecorder/session/prompting.mjs} +21 -21
  42. package/esm2020/lib/speechrecorder/session/recorder_combi_pane.mjs +75 -0
  43. package/esm2020/lib/speechrecorder/session/recording_list.mjs +109 -0
  44. package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +126 -0
  45. package/{esm2015/lib/speechrecorder/session/recordingfile/recording-file-navi.component.js → esm2020/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs} +5 -5
  46. package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +168 -0
  47. package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +415 -0
  48. package/{esm2015/lib/speechrecorder/session/recordingfile/recording-file.js → esm2020/lib/speechrecorder/session/recordingfile/recording-file.mjs} +1 -1
  49. package/esm2020/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +251 -0
  50. package/{esm2015/lib/speechrecorder/session/session.js → esm2020/lib/speechrecorder/session/session.mjs} +0 -0
  51. package/{esm2015/lib/speechrecorder/session/session.service.js → esm2020/lib/speechrecorder/session/session.service.mjs} +4 -4
  52. package/{esm2015/lib/speechrecorder/session/session_finished_dialog.js → esm2020/lib/speechrecorder/session/session_finished_dialog.mjs} +4 -4
  53. package/esm2020/lib/speechrecorder/session/sessionmanager.mjs +1274 -0
  54. package/{esm2015/lib/speechrecorder/session/warning_bar.js → esm2020/lib/speechrecorder/session/warning_bar.mjs} +5 -5
  55. package/{esm2015/lib/speechrecorder/spruploader.js → esm2020/lib/speechrecorder/spruploader.mjs} +4 -4
  56. package/{esm2015/lib/speechrecorder/startstopsignal/startstopsignal.js → esm2020/lib/speechrecorder/startstopsignal/startstopsignal.mjs} +0 -0
  57. package/{esm2015/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.js → esm2020/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.mjs} +5 -5
  58. package/esm2020/lib/speechrecorderng.component.mjs +427 -0
  59. package/{esm2015/lib/speechrecorderng.module.js → esm2020/lib/speechrecorderng.module.mjs} +12 -11
  60. package/{esm2015/lib/spr.config.js → esm2020/lib/spr.config.mjs} +4 -4
  61. package/{esm2015/lib/spr.module.version.js → esm2020/lib/spr.module.version.mjs} +2 -2
  62. package/{esm2015/lib/ui/canvas_layer_comp.js → esm2020/lib/ui/canvas_layer_comp.mjs} +4 -4
  63. package/{esm2015/lib/ui/message_dialog.js → esm2020/lib/ui/message_dialog.mjs} +4 -4
  64. package/{esm2015/lib/ui/recordingitem_display.js → esm2020/lib/ui/recordingitem_display.mjs} +9 -9
  65. package/{esm2015/lib/utils/css_utils.js → esm2020/lib/utils/css_utils.mjs} +0 -0
  66. package/{esm2015/lib/utils/scrollIntoViewToBottom.js → esm2020/lib/utils/scrollIntoViewToBottom.mjs} +4 -4
  67. package/{esm2015/lib/utils/ua-parser.js → esm2020/lib/utils/ua-parser.mjs} +0 -0
  68. package/{esm2015/lib/utils/utils.js → esm2020/lib/utils/utils.mjs} +0 -0
  69. package/esm2020/public-api.mjs +31 -0
  70. package/{esm2015/speechrecorderng.js → esm2020/speechrecorderng.mjs} +0 -0
  71. package/fesm2015/speechrecorderng.mjs +11991 -0
  72. package/fesm2015/speechrecorderng.mjs.map +1 -0
  73. package/{fesm2015/speechrecorderng.js → fesm2020/speechrecorderng.mjs} +1863 -285
  74. package/fesm2020/speechrecorderng.mjs.map +1 -0
  75. package/lib/media/utils.d.ts +3 -0
  76. package/lib/net/uploader.d.ts +4 -2
  77. package/lib/speechrecorder/recording.d.ts +14 -3
  78. package/lib/speechrecorder/recordings/recordings.service.d.ts +12 -5
  79. package/lib/speechrecorder/session/audiorecorder.d.ts +150 -0
  80. package/lib/speechrecorder/session/controlpanel.d.ts +2 -1
  81. package/lib/speechrecorder/session/item.d.ts +2 -2
  82. package/lib/speechrecorder/session/recorder_combi_pane.d.ts +26 -0
  83. package/lib/speechrecorder/session/recording_list.d.ts +19 -0
  84. package/lib/speechrecorder/session/recordingfile/recording-file-meta.component.d.ts +6 -2
  85. package/lib/speechrecorder/session/recordingfile/recording-file-view.component.d.ts +5 -4
  86. package/lib/speechrecorder/session/recordingfile/recording-file.d.ts +2 -2
  87. package/lib/speechrecorder/session/recordingfile/recordingfile-service.d.ts +5 -3
  88. package/lib/speechrecorder/session/sessionmanager.d.ts +5 -5
  89. package/lib/speechrecorderng.module.d.ts +31 -28
  90. package/lib/spr.module.version.d.ts +1 -1
  91. package/package.json +33 -20
  92. package/public-api.d.ts +2 -1
  93. package/bundles/speechrecorderng.umd.js +0 -10146
  94. package/bundles/speechrecorderng.umd.js.map +0 -1
  95. package/esm2015/lib/audio/audio_player.js +0 -274
  96. package/esm2015/lib/audio/capture/capture.js +0 -544
  97. package/esm2015/lib/audio/dsp/level_measure.js +0 -356
  98. package/esm2015/lib/audio/ui/audio_canvas_layer_comp.js +0 -364
  99. package/esm2015/lib/audio/ui/container.js +0 -412
  100. package/esm2015/lib/net/uploader.js +0 -220
  101. package/esm2015/lib/speechrecorder/recording.js +0 -38
  102. package/esm2015/lib/speechrecorder/recordings/recordings.service.js +0 -173
  103. package/esm2015/lib/speechrecorder/session/controlpanel.js +0 -446
  104. package/esm2015/lib/speechrecorder/session/recordingfile/recording-file-meta.component.js +0 -83
  105. package/esm2015/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.js +0 -163
  106. package/esm2015/lib/speechrecorder/session/recordingfile/recording-file-view.component.js +0 -386
  107. package/esm2015/lib/speechrecorder/session/recordingfile/recordingfile-service.js +0 -191
  108. package/esm2015/lib/speechrecorder/session/sessionmanager.js +0 -1271
  109. package/esm2015/lib/speechrecorderng.component.js +0 -423
  110. package/esm2015/public-api.js +0 -30
  111. package/fesm2015/speechrecorderng.js.map +0 -1
@@ -0,0 +1,547 @@
1
+ import { Browser, Platform, UserAgentBuilder } from "../../utils/ua-parser";
2
+ import { Platform as CfgPlatform } from "../../speechrecorder/project/project";
3
+ export const CHROME_ACTIVATE_ECHO_CANCELLATION_WITH_AGC = false;
4
+ const DEBUG_TRACE_LEVEL = 0;
5
+ const ENABLE_AUDIO_WORKLET = true;
6
+ // Super dirty way to load this module
7
+ // Copy content of interceptor_worklet.js to this string
8
+ const awpStr = "class AudioCaptureInterceptorProcessor extends AudioWorkletProcessor{\n" +
9
+ "\n" +
10
+ " BUFFER_QUANTUMS=64;\n" +
11
+ " QUANTUM_FRAME_LEN=128;\n" +
12
+ " BUFFER_FRAME_LEN=this.QUANTUM_FRAME_LEN*this.BUFFER_QUANTUMS;\n" +
13
+ " buffer=null;\n" +
14
+ " bufferPos=0;\n" +
15
+ " bufferPosBytes=0;\n" +
16
+ " constructor() {\n" +
17
+ " super();\n" +
18
+ "\n" +
19
+ " }\n" +
20
+ "\n" +
21
+ " process(\n" +
22
+ " inputs,\n" +
23
+ " outputs,\n" +
24
+ " parameters\n" +
25
+ " ){\n" +
26
+ "\n" +
27
+ " let inputsCnt=inputs.length;\n" +
28
+ " let channelCount=0;\n" +
29
+ " let inputLen=0;\n" +
30
+ " let inputLenBytes=0;\n" +
31
+ " if(inputsCnt>0) {\n" +
32
+ " let input0 = inputs[0];\n" +
33
+ " channelCount = input0.length;\n" +
34
+ " if (channelCount > 0) {\n" +
35
+ " let input0ch0=input0[0];\n" +
36
+ " inputLen=input0ch0.length;\n" +
37
+ " inputLenBytes=input0ch0.buffer.length;\n" +
38
+ " }\n" +
39
+ " }\n" +
40
+ " if (!this.buffer || this.buffer.length < channelCount) {\n" +
41
+ " this.buffer = new Array(channelCount);\n" +
42
+ " this.bufferPos = 0\n" +
43
+ " for (let bch = 0; bch < channelCount; bch++) {\n" +
44
+ " this.buffer[bch] = new Float32Array(this.BUFFER_FRAME_LEN);\n" +
45
+ " this.bufferPos = 0;\n" +
46
+ " this.bufferPosBytes=0;\n" +
47
+ " }\n" +
48
+ " }\n" +
49
+ " let bufAvail = this.BUFFER_FRAME_LEN - this.bufferPos;\n" +
50
+ " // check if buffer has to be transferred\n" +
51
+ " if (inputLen > bufAvail) {\n" +
52
+ " let ada=new Array(channelCount);\n" +
53
+ " for (let ch = 0; ch < channelCount; ch++) {\n" +
54
+ " ada[ch]=this.buffer[ch].buffer.slice(0);\n" +
55
+ " }\n" +
56
+ " this.port.postMessage({\n" +
57
+ " data: ada,\n" +
58
+ " chs: channelCount,\n" +
59
+ " len: this.bufferPos\n" +
60
+ " }, ada);\n" +
61
+ " // buffer transferred, reset\n" +
62
+ " this.bufferPos = 0;\n" +
63
+ " this.bufferPosBytes=0;\n" +
64
+ " }\n" +
65
+ "\n" +
66
+ " for(let ii=0;ii<inputsCnt;ii++) {\n" +
67
+ " for (let ch = 0; ch < channelCount; ch++) {\n" +
68
+ " // Mute outputs\n" +
69
+ " //outputs[ii][ch].fill(0);\n" +
70
+ " let chSamples = inputs[ii][ch];\n" +
71
+ " this.buffer[ch].set(chSamples,this.bufferPos);\n" +
72
+ " }\n" +
73
+ " this.bufferPos+=inputLen;\n" +
74
+ " this.bufferPosBytes+=inputLenBytes;\n" +
75
+ " }\n" +
76
+ " \n" +
77
+ " return true;\n" +
78
+ " }\n" +
79
+ "}\n" +
80
+ "\n" +
81
+ "registerProcessor('capture-interceptor',AudioCaptureInterceptorProcessor);\n";
82
+ export class AudioCapture {
83
+ constructor(context) {
84
+ this.agcStatus = null;
85
+ this.bufferingNode = null;
86
+ this.audioOutStream = null;
87
+ this.disconnectStreams = true;
88
+ this._opened = false;
89
+ this.capturing = false;
90
+ this.framesRecorded = 0;
91
+ this.context = context;
92
+ this.n = navigator;
93
+ }
94
+ get opened() {
95
+ return this._opened;
96
+ }
97
+ initData() {
98
+ this.data = new Array();
99
+ for (let i = 0; i < this.channelCount; i++) {
100
+ this.data.push(new Array());
101
+ }
102
+ this.framesRecorded = 0;
103
+ }
104
+ listDevices() {
105
+ navigator.mediaDevices.enumerateDevices().then((l) => this.printDevices(l));
106
+ }
107
+ dummySession() {
108
+ // workaround to request permissions:
109
+ // Start a dummy session
110
+ let mediaStrCnstrs = { audio: { echoCancelation: false }
111
+ };
112
+ return navigator.mediaDevices.getUserMedia(mediaStrCnstrs);
113
+ }
114
+ stopAllSessionTracks(mediaStream) {
115
+ let ats = mediaStream.getTracks();
116
+ for (let atIdx = 0; atIdx < ats.length; atIdx++) {
117
+ //console.debug("Stop dummy session track: #" + atIdx)
118
+ ats[atIdx].stop();
119
+ }
120
+ }
121
+ deviceInfos(cb, retry = true, dummyStream) {
122
+ navigator.mediaDevices.enumerateDevices().then((l) => {
123
+ let labelsAvailable = false;
124
+ for (let i = 0; i < l.length; i++) {
125
+ let di = l[i];
126
+ if (di.label) {
127
+ labelsAvailable = true;
128
+ }
129
+ }
130
+ if (!labelsAvailable) {
131
+ //console.debug("Media device enumeration: No labels.")
132
+ if (retry) {
133
+ console.info("Starting dummy session to request audio permissions...");
134
+ this.dummySession().then((s) => {
135
+ // and stop it immediately
136
+ if (s) {
137
+ //console.debug("Got dummy session stream: " + s + " .")
138
+ }
139
+ else {
140
+ //console.debug("No dummy stream")
141
+ }
142
+ // retry (only once)
143
+ this.deviceInfos(cb, false, s);
144
+ }, reason => {
145
+ //console.debug("Dummy session rejected.")
146
+ // TODO error callback
147
+ cb(null);
148
+ });
149
+ }
150
+ else {
151
+ cb(null);
152
+ }
153
+ }
154
+ else {
155
+ // success
156
+ cb(l);
157
+ }
158
+ if (dummyStream) {
159
+ this.stopAllSessionTracks(dummyStream);
160
+ }
161
+ }, (reason) => {
162
+ //rejected
163
+ //console.debug("Media device enumeration rejected.")
164
+ if (retry) {
165
+ //console.debug("Starting dummy session to request audio permissions...")
166
+ this.dummySession().then((s) => {
167
+ // and stop it immediately
168
+ //console.debug("Dummy session.")
169
+ if (s) {
170
+ //console.debug("Got dummy session stream: " + s + " .")
171
+ }
172
+ else {
173
+ //console.debug("No dummy stream")
174
+ }
175
+ // retry (only once)
176
+ this.deviceInfos(cb, false, s);
177
+ }, reason => {
178
+ //console.debug("Dummy session rejected.")
179
+ // TODO error callback
180
+ cb(null);
181
+ });
182
+ }
183
+ else {
184
+ cb(null);
185
+ }
186
+ if (dummyStream) {
187
+ this.stopAllSessionTracks(dummyStream);
188
+ }
189
+ });
190
+ }
191
+ printDevices(l) {
192
+ let selDeviceId = '___dummy___';
193
+ for (let i = 0; i < l.length; i++) {
194
+ let di = l[i];
195
+ console.log("Audio device: Id: " + di.deviceId + " groupId: " + di.groupId + " label: " + di.label + " kind: " + di.kind);
196
+ }
197
+ }
198
+ open(channelCount, selDeviceId, autoGainControlConfigs) {
199
+ this.context.resume().then(() => {
200
+ this._open(channelCount, selDeviceId, autoGainControlConfigs);
201
+ });
202
+ }
203
+ _open(channelCount, selDeviceId, autoGainControlConfigs) {
204
+ this.channelCount = channelCount;
205
+ this.framesRecorded = 0;
206
+ //var msc = new AudioStreamConstr();
207
+ // var msc={};
208
+ //msc.video = false;
209
+ //msc.audio = true;
210
+ // Chrome and Firefox stereo channels are identical !!
211
+ // And even worse: The data coming from the source is already preprocessed on FF and Chrome. It contains DSP artifacts!!
212
+ // https://bugs.chromium.org/p/chromium/issues/detail?id=387737
213
+ // The workaround to set these constraints does _NOT_ help:
214
+ //var msc={audio: {echoCancellation: false,channelCount: 2, googAudioMirroring: false},video: false};
215
+ // Safari at least version 11: Support for media streams
216
+ // TODO test if input is unprocessed
217
+ let msc;
218
+ console.info('User agent: ' + navigator.userAgent);
219
+ let ua = UserAgentBuilder.userAgent();
220
+ // ua.components.forEach((c)=>{
221
+ // console.info("UA_Comp: "+c.toString());
222
+ // })
223
+ let agcCfg = null;
224
+ let autoGainControl = false;
225
+ let chromeEchoCancellation = false;
226
+ if (autoGainControlConfigs) {
227
+ for (let agcc of autoGainControlConfigs) {
228
+ if (agcc.platform === CfgPlatform.Android && ua.detectedPlatform === Platform.Android) {
229
+ agcCfg = agcc;
230
+ break;
231
+ }
232
+ if (agcc.platform === CfgPlatform.Windows && ua.detectedPlatform === Platform.Windows) {
233
+ agcCfg = agcc;
234
+ break;
235
+ }
236
+ }
237
+ if (agcCfg) {
238
+ // TODO use EXACT/IDEAL constraint
239
+ autoGainControl = agcCfg.value;
240
+ if (CHROME_ACTIVATE_ECHO_CANCELLATION_WITH_AGC) {
241
+ chromeEchoCancellation = agcCfg.value;
242
+ }
243
+ // TODO query real AGC status
244
+ this.agcStatus = agcCfg.value;
245
+ }
246
+ else {
247
+ this.agcStatus = false;
248
+ }
249
+ }
250
+ // default
251
+ msc = {
252
+ audio: {
253
+ deviceId: selDeviceId,
254
+ echoCancellation: false,
255
+ channelCount: channelCount,
256
+ autoGainControl: autoGainControl
257
+ },
258
+ video: false
259
+ };
260
+ if (ua.detectedBrowser === Browser.Edge) {
261
+ // Microsoft Edge sends unmodified audio
262
+ // The constraint can follow the specification
263
+ console.info("Setting media track constraints for Microsoft Edge.");
264
+ msc = {
265
+ audio: {
266
+ deviceId: selDeviceId,
267
+ echoCancellation: false,
268
+ channelCount: channelCount,
269
+ autoGainControl: autoGainControl
270
+ },
271
+ video: false
272
+ };
273
+ }
274
+ else if (ua.detectedBrowser === Browser.Chrome) {
275
+ // Google Chrome: we need to switch of each of the preprocessing units including the
276
+ console.info("Setting media track constraints for Google Chrome.");
277
+ // Chrome 60 -> 61 changed
278
+ // it works now without mandatory/optional sub-objects
279
+ // Requires at least Chrome 61
280
+ msc = {
281
+ audio: {
282
+ deviceId: selDeviceId,
283
+ channelCount: channelCount,
284
+ echoCancellation: { exact: chromeEchoCancellation },
285
+ autoGainControl: { exact: autoGainControl },
286
+ sampleSize: { min: 16 },
287
+ },
288
+ video: false,
289
+ };
290
+ }
291
+ else if (ua.detectedBrowser === Browser.Firefox) {
292
+ console.info("Setting media track constraints for Mozilla Firefox.");
293
+ // Firefox
294
+ msc = {
295
+ audio: {
296
+ deviceId: selDeviceId,
297
+ channelCount: channelCount,
298
+ echoCancellation: false,
299
+ autoGainControl: autoGainControl,
300
+ noiseSuppression: false
301
+ },
302
+ video: false,
303
+ };
304
+ }
305
+ else if (ua.detectedBrowser === Browser.Safari) {
306
+ console.info("Setting media track constraints for Safari browser.");
307
+ console.info("Apply workaround for Safari: Avoid disconnect of streams.");
308
+ this.disconnectStreams = false;
309
+ msc = {
310
+ audio: {
311
+ deviceId: selDeviceId,
312
+ channelCount: channelCount,
313
+ echoCancellation: false
314
+ },
315
+ video: false,
316
+ };
317
+ }
318
+ else {
319
+ // TODO default constraints or error Browser not supported
320
+ }
321
+ console.debug("Audio capture, AGC: " + this.agcStatus);
322
+ let ump = navigator.mediaDevices.getUserMedia(msc);
323
+ ump.then((s) => {
324
+ this.stream = s;
325
+ let aTracks = s.getAudioTracks();
326
+ for (let i = 0; i < aTracks.length; i++) {
327
+ let aTrack = aTracks[i];
328
+ console.info("Track audio info: id: " + aTrack.id + " kind: " + aTrack.kind + " label: \"" + aTrack.label + "\"");
329
+ let mtrSts = aTrack.getSettings();
330
+ // Typescript lib.dom.ts MediaTrackSettings.channelCount is missing
331
+ // https://github.com/mdn/browser-compat-data/blob/5493d8f937e05b2ddbd41b99f5bdfad4a1f2ed85/api/MediaTrackSettings.json
332
+ //@ts-ignore
333
+ console.info("Track audio settings: Ch cnt: " + mtrSts.channelCount + ", AGC: " + mtrSts.autoGainControl + ", Echo cancell.: " + mtrSts.echoCancellation);
334
+ if (mtrSts.autoGainControl) {
335
+ this.agcStatus = mtrSts.autoGainControl;
336
+ }
337
+ }
338
+ let vTracks = s.getVideoTracks();
339
+ for (let i = 0; i < vTracks.length; i++) {
340
+ let vTrack = vTracks[i];
341
+ console.info("Track video info: id: " + vTrack.id + " kind: " + vTrack.kind + " label: " + vTrack.label);
342
+ }
343
+ this.mediaStream = this.context.createMediaStreamSource(s);
344
+ // stream channel count ( is always 2 !)
345
+ let streamChannelCount = this.mediaStream.channelCount;
346
+ console.info("Stream channel count: " + streamChannelCount);
347
+ // is not set!!
348
+ //this.currentSampleRate = this.mediaStream.sampleRate;
349
+ this.currentSampleRate = this.context.sampleRate;
350
+ console.info("Source audio node: channels: " + streamChannelCount + " samplerate: " + this.currentSampleRate);
351
+ if (this.audioOutStream) {
352
+ this.audioOutStream.setFormat(this.channelCount, this.currentSampleRate);
353
+ }
354
+ // W3C -> new name is createScriptProcessor
355
+ //
356
+ // Again deprecated, but AudioWorker not yet implemented in stable releases (June 2016)
357
+ // AudioWorker is now AudioWorkletProcessor ... (May 2017)
358
+ // Update 12-2020:
359
+ // The ScriptProcessorNode Interface - DEPRECATED
360
+ // Update 06-2021
361
+ // AudioWorkletProcessor is here to stay. Web Audio API has now Recommendation status !
362
+ if (ENABLE_AUDIO_WORKLET && this.context.audioWorklet) {
363
+ //const workletFileName = ('file-loader!./interceptor_worklet.js');
364
+ //const workletFileName = 'http://localhost:4200/assets/interceptor_worklet.js';
365
+ //console.log(awpStr);
366
+ let audioWorkletModuleBlob = new Blob([awpStr], { type: 'text/javascript' });
367
+ let audioWorkletModuleBlobUrl = window.URL.createObjectURL(audioWorkletModuleBlob);
368
+ this.context.audioWorklet.addModule(audioWorkletModuleBlobUrl).then(() => {
369
+ const awn = new AudioWorkletNode(this.context, 'capture-interceptor');
370
+ awn.onprocessorerror = (ev) => {
371
+ let msg = 'Unknwon error';
372
+ if (ev instanceof ErrorEvent) {
373
+ msg = ev.message;
374
+ }
375
+ console.error("Capture audio worklet error: " + msg);
376
+ if (this.listener) {
377
+ this.listener.error(msg);
378
+ }
379
+ };
380
+ let awnPt = awn.port;
381
+ if (awnPt) {
382
+ awnPt.onmessage = (ev) => {
383
+ if (this.capturing) {
384
+ let dt = ev.data;
385
+ let chs = dt.chs;
386
+ let adaLen = dt.data.length;
387
+ if (DEBUG_TRACE_LEVEL > 8) {
388
+ console.debug('Received data from worklet: ' + chs + ' ' + dt.len + ' Data chs: ' + adaLen);
389
+ }
390
+ //let chunkLen = adaLen / chs;
391
+ let chunkLen = adaLen;
392
+ let chunk = new Array(chs);
393
+ for (let ch = 0; ch < chs; ch++) {
394
+ if (this.data && this.data[ch]) {
395
+ let adaPos = ch * chunkLen;
396
+ if (dt.data[ch]) {
397
+ let fa = new Float32Array(dt.data[ch]);
398
+ this.data[ch].push(fa);
399
+ chunk[ch] = fa;
400
+ // Use samples of channel 0 to count frames (samples)
401
+ if (ch == 0) {
402
+ this.framesRecorded += fa.length;
403
+ }
404
+ }
405
+ else {
406
+ if (DEBUG_TRACE_LEVEL > 8) {
407
+ console.debug('Channel ' + ch + ' data not set!!');
408
+ }
409
+ }
410
+ }
411
+ }
412
+ if (this.audioOutStream) {
413
+ this.audioOutStream.write(chunk);
414
+ }
415
+ }
416
+ };
417
+ }
418
+ this.bufferingNode = awn;
419
+ this._opened = true;
420
+ if (this.listener) {
421
+ this.listener.opened();
422
+ }
423
+ }).catch((error) => {
424
+ console.log('Could not add module ' + error);
425
+ });
426
+ }
427
+ else if (this.context.createScriptProcessor) {
428
+ // The ScriptProcessorNode Interface - DEPRECATED Only as fallback
429
+ // TODO should we use streamChannelCount or channelCount here ?
430
+ let scriptProcessorNode = this.context.createScriptProcessor(AudioCapture.BUFFER_SIZE, streamChannelCount, streamChannelCount);
431
+ this.bufferingNode = scriptProcessorNode;
432
+ let c = 0;
433
+ if (scriptProcessorNode.onaudioprocess) {
434
+ scriptProcessorNode.onaudioprocess = (e) => {
435
+ if (this.capturing) {
436
+ let inBuffer = e.inputBuffer;
437
+ let duration = inBuffer.duration;
438
+ // only process requested count of channels
439
+ let currentBuffers = new Array(channelCount);
440
+ for (let ch = 0; ch < channelCount; ch++) {
441
+ let chSamples = inBuffer.getChannelData(ch);
442
+ let chSamplesCopy = chSamples.slice(0);
443
+ currentBuffers[ch] = chSamplesCopy.slice(0);
444
+ this.data[ch].push(chSamplesCopy);
445
+ if (DEBUG_TRACE_LEVEL > 8) {
446
+ console.debug("Process " + chSamplesCopy.length + " samples.");
447
+ }
448
+ this.framesRecorded += chSamplesCopy.length;
449
+ }
450
+ c++;
451
+ if (this.audioOutStream) {
452
+ this.audioOutStream.write(currentBuffers);
453
+ }
454
+ }
455
+ };
456
+ this._opened = true;
457
+ if (this.listener) {
458
+ this.listener.opened();
459
+ }
460
+ }
461
+ else {
462
+ this.listener.error('Browser does not support audio processing (ScriptProcessor.onaudioprocess method not found)!');
463
+ }
464
+ }
465
+ else {
466
+ this.listener.error('Browser does not support audio processing (neither AudioWorkletProcessor nor ScriptProcessor)!');
467
+ }
468
+ }, (e) => {
469
+ console.error(e + " Error name: " + e.name);
470
+ if (this.listener) {
471
+ if ('NotAllowedError' === e.name) {
472
+ this.listener.error('Not allowed to use your microphone.', 'Please make sure that microphone access is allowed for this web page and reload the page.');
473
+ }
474
+ else if ('NotReadableError' === e.name) {
475
+ this.listener.error('Could not read from your audio device.', 'Please make sure your audio device is working.');
476
+ }
477
+ else if ('OverconstrainedError' === e.name) {
478
+ let eMsg = e.msg ? e.msg : 'Overconstrained media device request error.';
479
+ this.listener.error(eMsg);
480
+ }
481
+ else {
482
+ this.listener.error();
483
+ }
484
+ }
485
+ });
486
+ }
487
+ start() {
488
+ this.initData();
489
+ if (this.audioOutStream) {
490
+ this.audioOutStream.nextStream();
491
+ }
492
+ this.capturing = true;
493
+ if (this.bufferingNode) {
494
+ this.mediaStream.connect(this.bufferingNode);
495
+ this.bufferingNode.connect(this.context.destination);
496
+ }
497
+ if (this.listener) {
498
+ this.listener.started();
499
+ }
500
+ }
501
+ stop() {
502
+ if (this.disconnectStreams && this.bufferingNode) {
503
+ this.mediaStream.disconnect(this.bufferingNode);
504
+ this.bufferingNode.disconnect(this.context.destination);
505
+ }
506
+ if (this.audioOutStream) {
507
+ this.audioOutStream.flush();
508
+ }
509
+ this.capturing = false;
510
+ if (this.listener) {
511
+ this.listener.stopped();
512
+ }
513
+ }
514
+ close() {
515
+ this.mediaStream.disconnect();
516
+ if (this.stream) {
517
+ //this.stream.stop();
518
+ //'MediaStream.stop()' is deprecated and will be removed in M47, around November 2015. Please use 'MediaStream.active' instead.
519
+ //this.stream.active=false;
520
+ var mts = this.stream.getTracks();
521
+ for (var i = 0; i < mts.length; i++) {
522
+ mts[i].stop();
523
+ }
524
+ }
525
+ this._opened = false;
526
+ }
527
+ audioBuffer() {
528
+ var frameLen = 0;
529
+ var ch0Data = this.data[0];
530
+ for (var ch0Chk of ch0Data) {
531
+ frameLen += ch0Chk.length;
532
+ }
533
+ var ab = this.context.createBuffer(this.channelCount, frameLen, this.context.sampleRate);
534
+ for (var ch = 0; ch < this.channelCount; ch++) {
535
+ var chD = ab.getChannelData(ch);
536
+ var pos = 0;
537
+ for (var chChk of this.data[ch]) {
538
+ var bufLen = chChk.length;
539
+ chD.set(chChk, pos);
540
+ pos += bufLen;
541
+ }
542
+ }
543
+ return ab;
544
+ }
545
+ }
546
+ AudioCapture.BUFFER_SIZE = 8192;
547
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FwdHVyZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3NwZWVjaHJlY29yZGVybmcvc3JjL2xpYi9hdWRpby9jYXB0dXJlL2NhcHR1cmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFDLE9BQU8sRUFBRSxRQUFRLEVBQUUsZ0JBQWdCLEVBQUMsTUFBTSx1QkFBdUIsQ0FBQztBQUUxRSxPQUFPLEVBQXdCLFFBQVEsSUFBSSxXQUFXLEVBQUMsTUFBTSxzQ0FBc0MsQ0FBQztBQUdwRyxNQUFNLENBQUMsTUFBTSwwQ0FBMEMsR0FBQyxLQUFLLENBQUM7QUFFOUQsTUFBTSxpQkFBaUIsR0FBQyxDQUFDLENBQUM7QUFFMUIsTUFBTSxvQkFBb0IsR0FBQyxJQUFJLENBQUM7QUFFaEMsc0NBQXNDO0FBQ3RDLHdEQUF3RDtBQUN4RCxNQUFNLE1BQU0sR0FBQyx5RUFBeUU7SUFDbEYsSUFBSTtJQUNKLDJCQUEyQjtJQUMzQiw4QkFBOEI7SUFDOUIscUVBQXFFO0lBQ3JFLG9CQUFvQjtJQUNwQixvQkFBb0I7SUFDcEIseUJBQXlCO0lBQ3pCLHVCQUF1QjtJQUN2QixvQkFBb0I7SUFDcEIsSUFBSTtJQUNKLFNBQVM7SUFDVCxJQUFJO0lBQ0osYUFBYTtJQUNiLGlCQUFpQjtJQUNqQixrQkFBa0I7SUFDbEIsb0JBQW9CO0lBQ3BCLFFBQVE7SUFDUixJQUFJO0lBQ0oscUNBQXFDO0lBQ3JDLDRCQUE0QjtJQUM1Qix3QkFBd0I7SUFDeEIsNkJBQTZCO0lBQzdCLDBCQUEwQjtJQUMxQixvQ0FBb0M7SUFDcEMsMENBQTBDO0lBQzFDLG9DQUFvQztJQUNwQyx5Q0FBeUM7SUFDekMsMkNBQTJDO0lBQzNDLHVEQUF1RDtJQUN2RCxjQUFjO0lBQ2QsVUFBVTtJQUNWLGlFQUFpRTtJQUNqRSxtREFBbUQ7SUFDbkQsK0JBQStCO0lBQy9CLDJEQUEyRDtJQUMzRCw0RUFBNEU7SUFDNUUsb0NBQW9DO0lB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LENBQUM7SUFFRCxXQUFXO1FBQ1QsSUFBSSxRQUFRLEdBQVcsQ0FBQyxDQUFDO1FBQ3pCLElBQUksT0FBTyxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFM0IsS0FBSyxJQUFJLE1BQU0sSUFBSSxPQUFPLEVBQUU7WUFDMUIsUUFBUSxJQUFJLE1BQU0sQ0FBQyxNQUFNLENBQUM7U0FDM0I7UUFDRCxJQUFJLEVBQUUsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsWUFBWSxFQUFFLFFBQVEsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxDQUFDO1FBQ3pGLEtBQUssSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLEVBQUUsR0FBRyxJQUFJLENBQUMsWUFBWSxFQUFFLEVBQUUsRUFBRSxFQUFFO1lBRTdDLElBQUksR0FBRyxHQUFHLEVBQUUsQ0FBQyxjQUFjLENBQUMsRUFBRSxDQUFDLENBQUM7WUFDaEMsSUFBSSxHQUFHLEdBQUcsQ0FBQyxDQUFDO1lBQ1osS0FBSyxJQUFJLEtBQUssSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxFQUFFO2dCQUMvQixJQUFJLE1BQU0sR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDO2dCQUMxQixHQUFHLENBQUMsR0FBRyxDQUFDLEtBQUssRUFBRSxHQUFHLENBQUMsQ0FBQztnQkFDcEIsR0FBRyxJQUFJLE1BQU0sQ0FBQzthQUNmO1NBQ0Y7UUFDRCxPQUFPLEVBQUUsQ0FBQztJQUNaLENBQUM7O0FBemdCTSx3QkFBVyxHQUFXLElBQUksQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7U2VxdWVuY2VBdWRpb0Zsb2F0MzJPdXRTdHJlYW19IGZyb20gXCIuLi9pby9zdHJlYW1cIjtcclxuaW1wb3J0IHtCcm93c2VyLCBQbGF0Zm9ybSwgVXNlckFnZW50QnVpbGRlcn0gZnJvbSBcIi4uLy4uL3V0aWxzL3VhLXBhcnNlclwiO1xyXG5cclxuaW1wb3J0IHtBdXRvR2FpbkNvbnRyb2xDb25maWcsIFBsYXRmb3JtIGFzIENmZ1BsYXRmb3JtfSBmcm9tIFwiLi4vLi4vc3BlZWNocmVjb3JkZXIvcHJvamVjdC9wcm9qZWN0XCI7XHJcblxyXG5cclxuZXhwb3J0IGNvbnN0IENIUk9NRV9BQ1RJVkFURV9FQ0hPX0NBTkNFTExBVElPTl9XSVRIX0FHQz1mYWxzZTtcclxuXHJcbmNvbnN0IERFQlVHX1RSQUNFX0xFVkVMPTA7XHJcblxyXG5jb25zdCBFTkFCTEVfQVVESU9fV09SS0xFVD10cnVlO1xyXG5cclxuLy8gU3VwZXIgZGlydHkgd2F5IHRvIGxvYWQgdGhpcyBtb2R1bGVcclxuLy8gQ29weSBjb250ZW50IG9mIGludGVyY2VwdG9yX3dvcmtsZXQuanMgdG8gdGhpcyBzdHJpbmdcclxuY29uc3QgYXdwU3RyPVwiY2xhc3MgQXVkaW9DYXB0dXJlSW50ZXJjZXB0b3JQcm9jZXNzb3IgZXh0ZW5kcyBBdWRpb1dvcmtsZXRQcm9jZXNzb3J7XFxuXCIgK1xyXG4gICAgXCJcXG5cIiArXHJcbiAgICBcIiAgICBCVUZGRVJfUVVBTlRVTVM9NjQ7XFxuXCIgK1xyXG4gICAgXCIgICAgUVVBTlRVTV9GUkFNRV9MRU49MTI4O1xcblwiICtcclxuICAgIFwiICAgIEJVRkZFUl9GUkFNRV9MRU49dGhpcy5RVUFOVFVNX0ZSQU1FX0xFTip0aGlzLkJVRkZFUl9RVUFOVFVNUztcXG5cIiArXHJcbiAgICBcIiAgICBidWZmZXI9bnVsbDtcXG5cIiArXHJcbiAgICBcIiAgICBidWZmZXJQb3M9MDtcXG5cIiArXHJcbiAgICBcIiAgICBidWZmZXJQb3NCeXRlcz0wO1xcblwiICtcclxuICAgIFwiICAgIGNvbnN0cnVjdG9yKCkge1xcblwiICtcclxuICAgIFwiICAgICAgICBzdXBlcigpO1xcblwiICtcclxuICAgIFwiXFxuXCIgK1xyXG4gICAgXCIgICAgfVxcblwiICtcclxuICAgIFwiXFxuXCIgK1xyXG4gICAgXCIgcHJvY2VzcyhcXG5cIiArXHJcbiAgICBcIiAgICAgIGlucHV0cyxcXG5cIiArXHJcbiAgICBcIiAgICAgIG91dHB1dHMsXFxuXCIgK1xyXG4gICAgXCIgICAgICBwYXJhbWV0ZXJzXFxuXCIgK1xyXG4gICAgXCIgICl7XFxuXCIgK1xyXG4gICAgXCJcXG5cIiArXHJcbiAgICBcIiAgICAgbGV0IGlucHV0c0NudD1pbnB1dHMubGVuZ3RoO1xcblwiICtcclxuICAgIFwiICAgICBsZXQgY2hhbm5lbENvdW50PTA7XFxuXCIgK1xyXG4gICAgXCIgICAgIGxldCBpbnB1dExlbj0wO1xcblwiICtcclxuICAgIFwiICAgICBsZXQgaW5wdXRMZW5CeXRlcz0wO1xcblwiICtcclxuICAgIFwiICAgICBpZihpbnB1dHNDbnQ+MCkge1xcblwiICtcclxuICAgIFwiICAgICAgICAgbGV0IGlucHV0MCA9IGlucHV0c1swXTtcXG5cIiArXHJcbiAgICBcIiAgICAgICAgIGNoYW5uZWxDb3VudCA9IGlucHV0MC5sZW5ndGg7XFxuXCIgK1xyXG4gICAgXCIgICAgICAgICBpZiAoY2hhbm5lbENvdW50ID4gMCkge1xcblwiICtcclxuICAgIFwiICAgICAgICAgICAgIGxldCBpbnB1dDBjaDA9aW5wdXQwWzBdO1xcblwiICtcclxuICAgIFwiICAgICAgICAgICAgIGlucHV0TGVuPWlucHV0MGNoMC5sZW5ndGg7XFxuXCIgK1xyXG4gICAgXCIgICAgICAgICAgICAgaW5wdXRMZW5CeXRlcz1pbnB1dDBjaDAuYnVmZmVyLmxlbmd0aDtcXG5cIiArXHJcbiAgICBcIiAgICAgICAgIH1cXG5cIiArXHJcbiAgICBcIiAgICAgfVxcblwiICtcclxuICAgIFwiICAgICBpZiAoIXRoaXMuYnVmZmVyIHx8IHRoaXMuYnVmZmVyLmxlbmd0aCA8IGNoYW5uZWxDb3VudCkge1xcblwiICtcclxuICAgIFwiICAgICAgICAgdGhpcy5idWZmZXIgPSBuZXcgQXJyYXkoY2hhbm5lbENvdW50KTtcXG5cIiArXHJcbiAgICBcIiAgICAgICAgIHRoaXMuYnVmZmVyUG9zID0gMFxcblwiICtcclxuICAgIFwiICAgICAgICAgZm9yIChsZXQgYmNoID0gMDsgYmNoIDwgY2hhbm5lbENvdW50OyBiY2grKykge1xcblwiICtcclxuICAgIFwiICAgICAgICAgICAgIHRoaXMuYnVmZmVyW2JjaF0gPSBuZXcgRmxvYXQzMkFycmF5KHRoaXMuQlVGRkVSX0ZSQU1FX0xFTik7XFxuXCIgK1xyXG4gICAgXCIgICAgICAgICAgICAgdGhpcy5idWZmZXJQb3MgPSAwO1xcblwiICtcclxuICAgIFwiICAgICAgICAgICAgIHRoaXMuYnVmZmVyUG9zQnl0ZXM9MDtcXG5cIiArXHJcbiAgICBcIiAgICAgICAgIH1cXG5cIiArXHJcbiAgICBcIiAgICAgfVxcblwiICtcclxuICAgIFwiICAgICBsZXQgYnVmQXZhaWwgPSB0aGlzLkJVRkZFUl9GUkFNRV9MRU4gLSB0aGlzLmJ1ZmZlclBvcztcXG5cIiArXHJcbiAgICBcIiAgICAgLy8gY2hlY2sgaWYgYnVmZmVyIGhhcyB0byBiZSB0cmFuc2ZlcnJlZFxcblwiICtcclxuICAgIFwiICAgICBpZiAoaW5wdXRMZW4gPiBidWZBdmFpbCkge1xcblwiICtcclxuICAgIFwiICAgICAgICAgbGV0IGFkYT1uZXcgQXJyYXkoY2hhbm5lbENvdW50KTtcXG5cIiArXHJcbiAgICBcIiAgICAgICAgIGZvciAobGV0IGNoID0gMDsgY2ggPCBjaGFubmVsQ291bnQ7IGNoKyspIHtcXG5cIiArXHJcbiAgICBcIiAgICAgICAgICAgICBhZGFbY2hdPXRoaXMuYnVmZmVyW2NoXS5idWZmZXIuc2xpY2UoMCk7XFxuXCIgK1xyXG4gICAgXCIgICAgICAgICB9XFxuXCIgK1xyXG4gICAgXCIgICAgICAgICB0aGlzLnBvcnQucG9zdE1lc3NhZ2Uoe1xcblwiICtcclxuICAgIFwiICAgICAgICAgICAgIGRhdGE6IGFkYSxcXG5cIiArXHJcbiAgICBcIiAgICAgICAgICAgICBjaHM6IGNoYW5uZWxDb3VudCxcXG5cIiArXHJcbiAgICBcIiAgICAgICAgICAgICBsZW46IHRoaXMuYnVmZmVyUG9zXFxuXCIgK1xyXG4gICAgXCIgICAgICAgICB9LCBhZGEpO1xcblwiICtcclxuICAgIFwiICAgICAgICAgLy8gYnVmZmVyIHRyYW5zZmVycmVkLCByZXNldFxcblwiICtcclxuICAgIFwiICAgICAgICAgdGhpcy5idWZmZXJQb3MgPSAwO1xcblwiICtcclxuICAgIFwiICAgICAgICAgdGhpcy5idWZmZXJQb3NCeXRlcz0wO1xcblwiICtcclxuICAgIFwiICAgICB9XFxuXCIgK1xyXG4gICAgXCJcXG5cIiArXHJcbiAgICBcIiAgICAgZm9yKGxldCBpaT0wO2lpPGlucHV0c0NudDtpaSsrKSB7XFxuXCIgK1xyXG4gICAgXCIgICAgICAgICBmb3IgKGxldCBjaCA9IDA7IGNoIDwgY2hhbm5lbENvdW50OyBjaCsrKSB7XFxuXCIgK1xyXG4gICAgXCIgICAgICAgICAgICAgLy8gTXV0ZSBvdXRwdXRzXFxuXCIgK1xyXG4gICAgXCIgICAgICAgICAgICAgLy9vdXRwdXRzW2lpXVtjaF0uZmlsbCgwKTtcXG5cIiArXHJcbiAgICBcIiAgICAgICAgICAgICBsZXQgY2hTYW1wbGVzID0gaW5wdXRzW2lpXVtjaF07XFxuXCIgK1xyXG4gICAgXCIgICAgICAgICAgICAgdGhpcy5idWZmZXJbY2hdLnNldChjaFNhbXBsZXMsdGhpcy5idWZmZXJQb3MpO1xcblwiICtcclxuICAgIFwiICAgICAgICAgfVxcblwiICtcclxuICAgIFwiICAgICAgICAgdGhpcy5idWZmZXJQb3MrPWlucHV0TGVuO1xcblwiICtcclxuICAgIFwiICAgICAgICAgdGhpcy5idWZmZXJQb3NCeXRlcys9aW5wdXRMZW5CeXRlcztcXG5cIiArXHJcbiAgICBcIiAgICAgfVxcblwiICtcclxuICAgIFwiICAgIFxcblwiICtcclxuICAgIFwiICAgICByZXR1cm4gdHJ1ZTtcXG5cIiArXHJcbiAgICBcIiAgfVxcblwiICtcclxuICAgIFwifVxcblwiICtcclxuICAgIFwiXFxuXCIgK1xyXG4gICAgXCJyZWdpc3RlclByb2Nlc3NvcignY2FwdHVyZS1pbnRlcmNlcHRvcicsQXVkaW9DYXB0dXJlSW50ZXJjZXB0b3JQcm9jZXNzb3IpO1xcblwiO1xyXG5cclxuXHJcblxyXG5cclxuZXhwb3J0IGludGVyZmFjZSBBdWRpb0NhcHR1cmVMaXN0ZW5lciB7XHJcbiAgb3BlbmVkKCk6IHZvaWQ7XHJcblxyXG4gIHN0YXJ0ZWQoKTogdm9pZDtcclxuXHJcbiAgc3RvcHBlZCgpOiB2b2lkO1xyXG5cclxuICBjbG9zZWQoKTogdm9pZDtcclxuXHJcbiAgZXJyb3IobXNnPzpzdHJpbmcsYWR2aWNlPzpzdHJpbmcpOiB2b2lkO1xyXG59XHJcblxyXG5leHBvcnQgY2xhc3MgQXVkaW9DYXB0dXJlIHtcclxuICBnZXQgb3BlbmVkKCk6IGJvb2xlYW4ge1xyXG4gICAgcmV0dXJuIHRoaXMuX29wZW5lZDtcclxuICB9XHJcblxyXG4gIHN0YXRpYyBCVUZGRVJfU0laRTogbnVtYmVyID0gODE5MjtcclxuICBjb250ZXh0OiBBdWRpb0NvbnRleHQ7XHJcbiAgc3RyZWFtITogTWVkaWFTdHJlYW07XHJcbiAgY2hhbm5lbENvdW50ITogbnVtYmVyO1xyXG4gIG1lZGlhU3RyZWFtOiBhbnk7XHJcbiAgYWdjU3RhdHVzOmJvb2xlYW58bnVsbD1udWxsO1xyXG4gIGJ1ZmZlcmluZ05vZGU6IEF1ZGlvTm9kZXxudWxsPW51bGw7XHJcbiAgbGlzdGVuZXIhOiBBdWRpb0NhcHR1cmVMaXN0ZW5lcjtcclxuICBkYXRhITogQXJyYXk8QXJyYXk8RmxvYXQzMkFycmF5Pj47XHJcbiAgY3VycmVudFNhbXBsZVJhdGUhOiBudW1iZXI7XHJcbiAgbjogTmF2aWdhdG9yO1xyXG4gIGF1ZGlvT3V0U3RyZWFtOiBTZXF1ZW5jZUF1ZGlvRmxvYXQzMk91dFN0cmVhbSB8IG51bGw9bnVsbDtcclxuICBwcml2YXRlIGRpc2Nvbm5lY3RTdHJlYW1zID0gdHJ1ZTtcclxuICBwcml2YXRlIF9vcGVuZWQ9ZmFsc2U7XHJcbiAgcHJpdmF0ZSBjYXB0dXJpbmcgPSBmYWxzZTtcclxuXHJcbiAgZnJhbWVzUmVjb3JkZWQ6IG51bWJlcj0wO1xyXG5cclxuICBjb25zdHJ1Y3Rvcihjb250ZXh0OiBBdWRpb0NvbnRleHQpIHtcclxuICAgIHRoaXMuY29udGV4dCA9IGNvbnRleHQ7XHJcbiAgICB0aGlzLm4gPSBuYXZpZ2F0b3I7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGluaXREYXRhKCkge1xyXG4gICAgdGhpcy5kYXRhID0gbmV3IEFycmF5PEFycmF5PEZsb2F0MzJBcnJheT4+KCk7XHJcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IHRoaXMuY2hhbm5lbENvdW50OyBpKyspIHtcclxuICAgICAgdGhpcy5kYXRhLnB1c2gobmV3IEFycmF5PEZsb2F0MzJBcnJheT4oKSk7XHJcbiAgICB9XHJcbiAgICB0aGlzLmZyYW1lc1JlY29yZGVkID0gMDtcclxuICB9XHJcblxyXG4gIGxpc3REZXZpY2VzKCkge1xyXG4gICAgbmF2aWdhdG9yLm1lZGlhRGV2aWNlcy5lbnVtZXJhdGVEZXZpY2VzKCkudGhlbigobDogTWVkaWFEZXZpY2VJbmZvW10pID0+IHRoaXMucHJpbnREZXZpY2VzKGwpKTtcclxuICB9XHJcblxyXG4gIHByaXZhdGUgZHVtbXlTZXNzaW9uKCk6UHJvbWlzZTxNZWRpYVN0cmVhbT57XHJcbiAgICAvLyB3b3JrYXJvdW5kIHRvIHJlcXVlc3QgcGVybWlzc2lvbnM6XHJcbiAgICAvLyBTdGFydCBhIGR1bW15IHNlc3Npb25cclxuICAgIGxldCBtZWRpYVN0ckNuc3RycyA9IDxNZWRpYVN0cmVhbUNvbnN0cmFpbnRzPnthdWRpbzpcclxuICAgICAgICB7ZWNob0NhbmNlbGF0aW9uOiBmYWxzZX1cclxuICAgIH07XHJcbiAgICByZXR1cm4gbmF2aWdhdG9yLm1lZGlhRGV2aWNlcy5nZXRVc2VyTWVkaWEobWVkaWFTdHJDbnN0cnMpO1xyXG5cclxuICB9XHJcblxyXG5cclxuICBwcml2YXRlIHN0b3BBbGxTZXNzaW9uVHJhY2tzKG1lZGlhU3RyZWFtOk1lZGlhU3RyZWFtKXtcclxuICAgICAgbGV0IGF0cyA9IG1lZGlhU3RyZWFtLmdldFRyYWNrcygpO1xyXG4gICAgICBmb3IgKGxldCBhdElkeCA9IDA7IGF0SWR4IDwgYXRzLmxlbmd0aDsgYXRJZHgrKykge1xyXG4gICAgICAgIC8vY29uc29sZS5kZWJ1ZyhcIlN0b3AgZHVtbXkgc2Vzc2lvbiB0cmFjazogI1wiICsgYXRJZHgpXHJcbiAgICAgICAgYXRzW2F0SWR4XS5zdG9wKCk7XHJcbiAgICAgIH1cclxuICB9XHJcblxyXG4gIGRldmljZUluZm9zKGNiOiAoZGV2aWNlSW5mb3M6IE1lZGlhRGV2aWNlSW5mb1tdIHwgbnVsbCkgPT4gYW55LCByZXRyeSA9IHRydWUsZHVtbXlTdHJlYW0/Ok1lZGlhU3RyZWFtKSB7XHJcblxyXG4gICAgbmF2aWdhdG9yLm1lZGlhRGV2aWNlcy5lbnVtZXJhdGVEZXZpY2VzKCkudGhlbigobDogTWVkaWFEZXZpY2VJbmZvW10pID0+IHtcclxuICAgICAgbGV0IGxhYmVsc0F2YWlsYWJsZSA9IGZhbHNlO1xyXG4gICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGwubGVuZ3RoOyBpKyspIHtcclxuICAgICAgICBsZXQgZGkgPSBsW2ldO1xyXG4gICAgICAgIGlmIChkaS5sYWJlbCkge1xyXG4gICAgICAgICAgbGFiZWxzQXZhaWxhYmxlID0gdHJ1ZTtcclxuICAgICAgICB9XHJcbiAgICAgIH1cclxuICAgICAgaWYgKCFsYWJlbHNBdmFpbGFibGUpIHtcclxuICAgICAgICAvL2NvbnNvbGUuZGVidWcoXCJNZWRpYSBkZXZpY2UgZW51bWVyYXRpb246IE5vIGxhYmVscy5cIilcclxuICAgICAgICBpZiAocmV0cnkpIHtcclxuICAgICAgICAgICAgY29uc29sZS5pbmZvKFwiU3RhcnRpbmcgZHVtbXkgc2Vzc2lvbiB0byByZXF1ZXN0IGF1ZGlvIHBlcm1pc3Npb25zLi4uXCIpXHJcblxyXG4gICAgICAgICAgICB0aGlzLmR1bW15U2Vzc2lvbigpLnRoZW4oKHM6IE1lZGlhU3RyZWFtKSA9PiB7XHJcbiAgICAgICAgICAgIC8vIGFuZCBzdG9wIGl0IGltbWVkaWF0ZWx5XHJcblxyXG4gICAgICAgICAgICBpZihzKSB7XHJcbiAgICAgICAgICAgICAgLy9jb25zb2xlLmRlYnVnKFwiR290IGR1bW15IHNlc3Npb24gc3RyZWFtOiBcIiArIHMgKyBcIiAuXCIpXHJcbiAgICAgICAgICAgIH1lbHNle1xyXG4gICAgICAgICAgICAgIC8vY29uc29sZS5kZWJ1ZyhcIk5vIGR1bW15IHN0cmVhbVwiKVxyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIC8vIHJldHJ5IChvbmx5IG9uY2UpXHJcbiAgICAgICAgICAgIHRoaXMuZGV2aWNlSW5mb3MoY2IsIGZhbHNlLHMpO1xyXG4gICAgICAgICAgfSxyZWFzb24gPT4ge1xyXG4gICAgICAgICAgICAvL2NvbnNvbGUuZGVidWcoXCJEdW1teSBzZXNzaW9uIHJlamVjdGVkLlwiKVxyXG4gICAgICAgICAgICAvLyBUT0RPIGVycm9yIGNhbGxiYWNrXHJcbiAgICAgICAgICAgIGNiKG51bGwpO1xyXG4gICAgICAgICAgfSk7XHJcbiAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgIGNiKG51bGwpO1xyXG4gICAgICAgIH1cclxuICAgICAgfSBlbHNlIHtcclxuICAgICAgICAvLyBzdWNjZXNzXHJcbiAgICAgICAgY2IobCk7XHJcbiAgICAgIH1cclxuICAgICAgaWYoZHVtbXlTdHJlYW0pe1xyXG4gICAgICAgIHRoaXMuc3RvcEFsbFNlc3Npb25UcmFja3MoZHVtbXlTdHJlYW0pO1xyXG4gICAgICB9XHJcbiAgICB9LChyZWFzb24pPT4ge1xyXG4gICAgICAvL3JlamVjdGVkXHJcbiAgICAgIC8vY29uc29sZS5kZWJ1ZyhcIk1lZGlhIGRldmljZSBlbnVtZXJhdGlvbiByZWplY3RlZC5cIilcclxuICAgICAgaWYgKHJldHJ5KSB7XHJcbiAgICAgICAgLy9jb25zb2xlLmRlYnVnKFwiU3RhcnRpbmcgZHVtbXkgc2Vzc2lvbiB0byByZXF1ZXN0IGF1ZGlvIHBlcm1pc3Npb25zLi4uXCIpXHJcbiAgICAgICAgdGhpcy5kdW1teVNlc3Npb24oKS50aGVuKChzOiBNZWRpYVN0cmVhbSkgPT4ge1xyXG4gICAgICAgICAgLy8gYW5kIHN0b3AgaXQgaW1tZWRpYXRlbHlcclxuICAgICAgICAgIC8vY29uc29sZS5kZWJ1ZyhcIkR1bW15IHNlc3Npb24uXCIpXHJcbiAgICAgICAgICBpZihzKSB7XHJcbiAgICAgICAgICAgIC8vY29uc29sZS5kZWJ1ZyhcIkdvdCBkdW1teSBzZXNzaW9uIHN0cmVhbTogXCIgKyBzICsgXCIgLlwiKVxyXG4gICAgICAgICAgfWVsc2V7XHJcbiAgICAgICAgICAgIC8vY29uc29sZS5kZWJ1ZyhcIk5vIGR1bW15IHN0cmVhbVwiKVxyXG4gICAgICAgICAgfVxyXG4gICAgICAgICAgLy8gcmV0cnkgKG9ubHkgb25jZSlcclxuICAgICAgICAgIHRoaXMuZGV2aWNlSW5mb3MoY2IsIGZhbHNlLHMpO1xyXG4gICAgICAgIH0sIHJlYXNvbiA9PiB7XHJcbiAgICAgICAgICAvL2NvbnNvbGUuZGVidWcoXCJEdW1teSBzZXNzaW9uIHJlamVjdGVkLlwiKVxyXG4gICAgICAgICAgLy8gVE9ETyBlcnJvciBjYWxsYmFja1xyXG4gICAgICAgICAgY2IobnVsbCk7XHJcbiAgICAgICAgfSk7XHJcbiAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgY2IobnVsbCk7XHJcbiAgICAgIH1cclxuICAgICAgaWYoZHVtbXlTdHJlYW0pe1xyXG4gICAgICAgIHRoaXMuc3RvcEFsbFNlc3Npb25UcmFja3MoZHVtbXlTdHJlYW0pO1xyXG4gICAgICB9XHJcbiAgICB9KTtcclxuXHJcblxyXG5cclxuICB9XHJcblxyXG5cclxuICBwcmludERldmljZXMobDogTWVkaWFEZXZpY2VJbmZvW10pOiB2b2lkIHtcclxuICAgIGxldCBzZWxEZXZpY2VJZCA9ICdfX19kdW1teV9fXyc7XHJcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGwubGVuZ3RoOyBpKyspIHtcclxuICAgICAgbGV0IGRpID0gbFtpXTtcclxuXHJcbiAgICAgIGNvbnNvbGUubG9nKFwiQXVkaW8gZGV2aWNlOiBJZDogXCIgKyBkaS5kZXZpY2VJZCArIFwiIGdyb3VwSWQ6IFwiICsgZGkuZ3JvdXBJZCArIFwiIGxhYmVsOiBcIiArIGRpLmxhYmVsICsgXCIga2luZDogXCIgKyBkaS5raW5kKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIG9wZW4oY2hhbm5lbENvdW50OiBudW1iZXIsIHNlbERldmljZUlkPzogQ29uc3RyYWluRE9NU3RyaW5nfHVuZGVmaW5lZCxhdXRvR2FpbkNvbnRyb2xDb25maWdzPzpBcnJheTxBdXRvR2FpbkNvbnRyb2xDb25maWc+fG51bGx8dW5kZWZpbmVkKXtcclxuICAgICAgdGhpcy5jb250ZXh0LnJlc3VtZSgpLnRoZW4oKCk9PntcclxuICAgICAgICB0aGlzLl9vcGVuKGNoYW5uZWxDb3VudCxzZWxEZXZpY2VJZCxhdXRvR2FpbkNvbnRyb2xDb25maWdzKTtcclxuICAgICAgfSlcclxuICB9XHJcblxyXG4gIF9vcGVuKGNoYW5uZWxDb3VudDogbnVtYmVyLCBzZWxEZXZpY2VJZD86IENvbnN0cmFpbkRPTVN0cmluZ3x1bmRlZmluZWQsYXV0b0dhaW5Db250cm9sQ29uZmlncz86QXJyYXk8QXV0b0dhaW5Db250cm9sQ29uZmlnPnxudWxsfHVuZGVmaW5lZCkge1xyXG4gICAgdGhpcy5jaGFubmVsQ291bnQgPSBjaGFubmVsQ291bnQ7XHJcbiAgICB0aGlzLmZyYW1lc1JlY29yZGVkID0gMDtcclxuICAgIC8vdmFyIG1zYyA9IG5ldyBBdWRpb1N0cmVhbUNvbnN0cigpO1xyXG4gICAgLy8gdmFyIG1zYz17fTtcclxuICAgIC8vbXNjLnZpZGVvID0gZmFsc2U7XHJcbiAgICAvL21zYy5hdWRpbyA9IHRydWU7XHJcblxyXG4gICAgLy8gQ2hyb21lIGFuZCBGaXJlZm94IHN0ZXJlbyBjaGFubmVscyBhcmUgaWRlbnRpY2FsICEhXHJcbiAgICAvLyBBbmQgZXZlbiB3b3JzZTogVGhlIGRhdGEgY29taW5nIGZyb20gdGhlIHNvdXJjZSBpcyBhbHJlYWR5IHByZXByb2Nlc3NlZCBvbiBGRiBhbmQgQ2hyb21lLiBJdCBjb250YWlucyBEU1AgYXJ0aWZhY3RzISFcclxuXHJcbiAgICAvLyBodHRwczovL2J1Z3MuY2hyb21pdW0ub3JnL3AvY2hyb21pdW0vaXNzdWVzL2RldGFpbD9pZD0zODc3MzdcclxuICAgIC8vIFRoZSB3b3JrYXJvdW5kIHRvIHNldCB0aGVzZSBjb25zdHJhaW50cyBkb2VzIF9OT1RfIGhlbHA6XHJcbiAgICAvL3ZhciBtc2M9e2F1ZGlvOiB7ZWNob0NhbmNlbGxhdGlvbjogZmFsc2UsY2hhbm5lbENvdW50OiAyLCBnb29nQXVkaW9NaXJyb3Jpbmc6IGZhbHNlfSx2aWRlbzogZmFsc2V9O1xyXG5cclxuICAgIC8vIFNhZmFyaSBhdCBsZWFzdCB2ZXJzaW9uIDExOiBTdXBwb3J0IGZvciBtZWRpYSBzdHJlYW1zXHJcbiAgICAvLyBUT0RPIHRlc3QgaWYgaW5wdXQgaXMgdW5wcm9jZXNzZWRcclxuXHJcbiAgICBsZXQgbXNjOk1lZGlhU3RyZWFtQ29uc3RyYWludHM7XHJcbiAgICBjb25zb2xlLmluZm8oJ1VzZXIgYWdlbnQ6ICcrbmF2aWdhdG9yLnVzZXJBZ2VudCk7XHJcblxyXG5cclxuICAgICAgbGV0IHVhPVVzZXJBZ2VudEJ1aWxkZXIudXNlckFnZW50KCk7XHJcblxyXG4gICAgICAvLyB1YS5jb21wb25lbnRzLmZvckVhY2goKGMpPT57XHJcbiAgICAgIC8vICAgY29uc29sZS5pbmZvKFwiVUFfQ29tcDogXCIrYy50b1N0cmluZygpKTtcclxuICAgICAgLy8gfSlcclxuXHJcbiAgICAgbGV0IGFnY0NmZzpBdXRvR2FpbkNvbnRyb2xDb25maWd8bnVsbD1udWxsO1xyXG5cclxuICAgIGxldCBhdXRvR2FpbkNvbnRyb2w9ZmFsc2U7XHJcbiAgICBsZXQgY2hyb21lRWNob0NhbmNlbGxhdGlvbj1mYWxzZTtcclxuICAgIGlmKGF1dG9HYWluQ29udHJvbENvbmZpZ3Mpe1xyXG4gICAgICBmb3IobGV0IGFnY2Mgb2YgYXV0b0dhaW5Db250cm9sQ29uZmlncyl7XHJcbiAgICAgICAgaWYoYWdjYy5wbGF0Zm9ybT09PUNmZ1BsYXRmb3JtLkFuZHJvaWQgJiYgdWEuZGV0ZWN0ZWRQbGF0Zm9ybT09PVBsYXRmb3JtLkFuZHJvaWQpe1xyXG4gICAgICAgICAgICBhZ2NDZmc9YWdjYztcclxuICAgICAgICAgICAgYnJlYWs7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGlmKGFnY2MucGxhdGZvcm09PT1DZmdQbGF0Zm9ybS5XaW5kb3dzICYmIHVhLmRldGVjdGVkUGxhdGZvcm09PT1QbGF0Zm9ybS5XaW5kb3dzKXtcclxuICAgICAgICAgIGFnY0NmZz1hZ2NjO1xyXG4gICAgICAgICAgYnJlYWs7XHJcbiAgICAgICAgfVxyXG4gICAgICB9XHJcbiAgICAgIGlmKGFnY0NmZyl7XHJcbiAgICAgICAgLy8gVE9ETyB1c2UgRVhBQ1QvSURFQUwgY29uc3RyYWludFxyXG4gICAgICAgIGF1dG9HYWluQ29udHJvbD1hZ2NDZmcudmFsdWU7XHJcbiAgICAgICAgaWYoQ0hST01FX0FDVElWQVRFX0VDSE9fQ0FOQ0VMTEFUSU9OX1dJVEhfQUdDKXtcclxuICAgICAgICAgIGNocm9tZUVjaG9DYW5jZWxsYXRpb249YWdjQ2ZnLnZhbHVlO1xyXG4gICAgICAgIH1cclxuICAgICAgICAvLyBUT0RPIHF1ZXJ5IHJlYWwgQUdDIHN0YXR1c1xyXG4gICAgICAgIHRoaXMuYWdjU3RhdHVzPWFnY0NmZy52YWx1ZTtcclxuICAgICAgfWVsc2V7XHJcbiAgICAgICAgdGhpcy5hZ2NTdGF0dXM9ZmFsc2U7XHJcbiAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICAvLyBkZWZhdWx0XHJcbiAgICBtc2MgPSB7XHJcbiAgICAgIGF1ZGlvOiB7XHJcbiAgICAgICAgZGV2aWNlSWQ6IHNlbERldmljZUlkLFxyXG4gICAgICAgIGVjaG9DYW5jZWxsYXRpb246IGZhbHNlLFxyXG4gICAgICAgIGNoYW5uZWxDb3VudDogY2hhbm5lbENvdW50LFxyXG4gICAgICAgIGF1dG9HYWluQ29udHJvbDogYXV0b0dhaW5Db250cm9sXHJcbiAgICAgIH0sXHJcbiAgICAgIHZpZGVvOiBmYWxzZVxyXG4gICAgfTtcclxuXHJcbiAgICBpZiAodWEuZGV0ZWN0ZWRCcm93c2VyPT09QnJvd3Nlci5FZGdlKSB7XHJcblxyXG4gICAgICAvLyBNaWNyb3NvZnQgRWRnZSBzZW5kcyB1bm1vZGlmaWVkIGF1ZGlvXHJcbiAgICAgIC8vIFRoZSBjb25zdHJhaW50IGNhbiBmb2xsb3cgdGhlIHNwZWNpZmljYXRpb25cclxuICAgICAgY29uc29sZS5pbmZvKFwiU2V0dGluZyBtZWRpYSB0cmFjayBjb25zdHJhaW50cyBmb3IgTWljcm9zb2Z0IEVkZ2UuXCIpO1xyXG4gICAgICBtc2MgPSB7XHJcbiAgICAgICAgYXVkaW86IHtcclxuICAgICAgICAgIGRldmljZUlkOiBzZWxEZXZpY2VJZCxcclxuICAgICAgICAgIGVjaG9DYW5jZWxsYXRpb246IGZhbHNlLFxyXG4gICAgICAgICAgY2hhbm5lbENvdW50OiBjaGFubmVsQ291bnQsXHJcbiAgICAgICAgICBhdXRvR2FpbkNvbnRyb2w6IGF1dG9HYWluQ29udHJvbFxyXG4gICAgICAgIH0sXHJcbiAgICAgICAgdmlkZW86IGZhbHNlXHJcbiAgICAgIH07XHJcbiAgICB9IGVsc2UgaWYgKHVhLmRldGVjdGVkQnJvd3Nlcj09PUJyb3dzZXIuQ2hyb21lKSB7XHJcbiAgICAgIC8vIEdvb2dsZSBDaHJvbWU6IHdlIG5lZWQgdG8gc3dpdGNoIG9mIGVhY2ggb2YgdGhlIHByZXByb2Nlc3NpbmcgdW5pdHMgaW5jbHVkaW5nIHRoZVxyXG4gICAgICBjb25zb2xlLmluZm8oXCJTZXR0aW5nIG1lZGlhIHRyYWNrIGNvbnN0cmFpbnRzIGZvciBHb29nbGUgQ2hyb21lLlwiKTtcclxuXHJcbiAgICAgIC8vIENocm9tZSA2MCAtPiA2MSBjaGFuZ2VkXHJcbiAgICAgIC8vIGl0IHdvcmtzIG5vdyB3aXRob3V0IG1hbmRhdG9yeS9vcHRpb25hbCBzdWItb2JqZWN0c1xyXG5cclxuXHJcbiAgICAgIC8vIFJlcXVpcmVzIGF0IGxlYXN0IENocm9tZSA2MVxyXG4gICAgICBtc2MgPSB7XHJcbiAgICAgICAgYXVkaW86IHtcclxuICAgICAgICAgIGRldmljZUlkOiBzZWxEZXZpY2VJZCxcclxuICAgICAgICAgIGNoYW5uZWxDb3VudDogY2hhbm5lbENvdW50LFxyXG4gICAgICAgICAgZWNob0NhbmNlbGxhdGlvbjoge2V4YWN0OmNocm9tZUVjaG9DYW5jZWxsYXRpb259LFxyXG4gICAgICAgICAgYXV0b0dhaW5Db250cm9sOiB7ZXhhY3Q6YXV0b0dhaW5Db250cm9sfSxcclxuICAgICAgICAgIHNhbXBsZVNpemU6e21pbjogMTZ9LFxyXG4gICAgICAgIH0sXHJcbiAgICAgICAgdmlkZW86IGZhbHNlLFxyXG4gICAgICB9XHJcblxyXG4gICAgfSBlbHNlIGlmICh1YS5kZXRlY3RlZEJyb3dzZXI9PT1Ccm93c2VyLkZpcmVmb3gpIHtcclxuICAgICAgY29uc29sZS5pbmZvKFwiU2V0dGluZyBtZWRpYSB0cmFjayBjb25zdHJhaW50cyBmb3IgTW96aWxsYSBGaXJlZm94LlwiKTtcclxuICAgICAgLy8gRmlyZWZveFxyXG4gICAgICBtc2MgPSB7XHJcbiAgICAgICAgYXVkaW86IHtcclxuICAgICAgICAgICAgZGV2aWNlSWQ6IHNlbERldmljZUlkLFxyXG4gICAgICAgICAgICBjaGFubmVsQ291bnQ6IGNoYW5uZWxDb3VudCxcclxuICAgICAgICAgIGVjaG9DYW5jZWxsYXRpb246IGZhbHNlLFxyXG4gICAgICAgICAgICBhdXRvR2FpbkNvbnRyb2w6IGF1dG9HYWluQ29udHJvbCxcclxuICAgICAgICAgIG5vaXNlU3VwcHJlc3Npb246IGZhbHNlXHJcbiAgICAgICAgfSxcclxuICAgICAgICB2aWRlbzogZmFsc2UsXHJcbiAgICAgIH1cclxuXHJcbiAgICB9IGVsc2UgaWYgKHVhLmRldGVjdGVkQnJvd3Nlcj09PUJyb3dzZXIuU2FmYXJpKSB7XHJcbiAgICAgIGNvbnNvbGUuaW5mbyhcIlNldHRpbmcgbWVkaWEgdHJhY2sgY29uc3RyYWludHMgZm9yIFNhZmFyaSBicm93c2VyLlwiKVxyXG4gICAgICBjb25zb2xlLmluZm8oXCJBcHBseSB3b3JrYXJvdW5kIGZvciBTYWZhcmk6IEF2b2lkIGRpc2Nvbm5lY3Qgb2Ygc3RyZWFtcy5cIik7XHJcblxyXG4gICAgICB0aGlzLmRpc2Nvbm5lY3RTdHJlYW1zID0gZmFsc2U7XHJcbiAgICAgIG1zYyA9IHtcclxuICAgICAgICBhdWRpbzoge1xyXG4gICAgICAgICAgZGV2aWNlSWQ6IHNlbERldmljZUlkLFxyXG4gICAgICAgICAgY2hhbm5lbENvdW50OiBjaGFubmVsQ291bnQsXHJcbiAgICAgICAgICBlY2hvQ2FuY2VsbGF0aW9uOiBmYWxzZVxyXG4gICAgICAgIH0sXHJcbiAgICAgICAgdmlkZW86IGZhbHNlLFxyXG4gICAgICB9XHJcblxyXG4gICAgfSBlbHNlIHtcclxuXHJcbiAgICAgIC8vIFRPRE8gZGVmYXVsdCBjb25zdHJhaW50cyBvciBlcnJvciBCcm93c2VyIG5vdCBzdXBwb3J0ZWRcclxuICAgIH1cclxuXHJcblxyXG5cclxuICAgIGNvbnNvbGUuZGVidWcoXCJBdWRpbyBjYXB0dXJlLCBBR0M6IFwiK3RoaXMuYWdjU3RhdHVzKVxyXG5cclxuXHJcbiAgICBsZXQgdW1wID0gbmF2aWdhdG9yLm1lZGlhRGV2aWNlcy5nZXRVc2VyTWVkaWEobXNjKTtcclxuICAgIHVtcC50aGVuKChzKSA9PiB7XHJcbiAgICAgICAgdGhpcy5zdHJlYW0gPSBzO1xyXG5cclxuICAgICAgICBsZXQgYVRyYWNrcyA9IHMuZ2V0QXVkaW9UcmFja3MoKTtcclxuXHJcbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCBhVHJhY2tzLmxlbmd0aDsgaSsrKSB7XHJcbiAgICAgICAgICBsZXQgYVRyYWNrID0gYVRyYWNrc1tpXTtcclxuXHJcbiAgICAgICAgICBjb25zb2xlLmluZm8oXCJUcmFjayBhdWRpbyBpbmZvOiBpZDogXCIgKyBhVHJhY2suaWQgKyBcIiBraW5kOiBcIiArIGFUcmFjay5raW5kICsgXCIgbGFiZWw6IFxcXCJcIiArIGFUcmFjay5sYWJlbCArIFwiXFxcIlwiKTtcclxuICAgICAgICAgIGxldCBtdHJTdHM9YVRyYWNrLmdldFNldHRpbmdzKCk7XHJcblxyXG4gICAgICAgICAgLy8gVHlwZXNjcmlwdCBsaWIuZG9tLnRzIE1lZGlhVHJhY2tTZXR0aW5ncy5jaGFubmVsQ291bnQgaXMgbWlzc2luZ1xyXG4gICAgICAgICAgLy8gaHR0cHM6Ly9naXRodWIuY29tL21kbi9icm93c2VyLWNvbXBhdC1kYXRhL2Jsb2IvNTQ5M2Q4ZjkzN2UwNWIyZGRiZDQxYjk5ZjViZGZhZDRhMWYyZWQ4NS9hcGkvTWVkaWFUcmFja1NldHRpbmdzLmpzb25cclxuICAgICAgICAgIC8vQHRzLWlnbm9yZVxyXG4gICAgICAgICAgY29uc29sZS5pbmZvKFwiVHJhY2sgYXVkaW8gc2V0dGluZ3M6IENoIGNudDogXCIrbXRyU3RzLmNoYW5uZWxDb3VudCtcIiwgQUdDOiBcIittdHJTdHMuYXV0b0dhaW5Db250cm9sK1wiLCBFY2hvIGNhbmNlbGwuOiBcIittdHJTdHMuZWNob0NhbmNlbGxhdGlvbik7XHJcbiAgICAgICAgICBpZihtdHJTdHMuYXV0b0dhaW5Db250cm9sKXtcclxuICAgICAgICAgICAgdGhpcy5hZ2NTdGF0dXM9bXRyU3RzLmF1dG9HYWluQ29udHJvbDtcclxuICAgICAgICAgIH1cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGxldCB2VHJhY2tzID0gcy5nZXRWaWRlb1RyYWNrcygpO1xyXG4gICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgdlRyYWNrcy5sZW5ndGg7IGkrKykge1xyXG4gICAgICAgICAgbGV0IHZUcmFjayA9IHZUcmFja3NbaV07XHJcbiAgICAgICAgICBjb25zb2xlLmluZm8oXCJUcmFjayB2aWRlbyBpbmZvOiBpZDogXCIgKyB2VHJhY2suaWQgKyBcIiBraW5kOiBcIiArIHZUcmFjay5raW5kICsgXCIgbGFiZWw6IFwiICsgdlRyYWNrLmxhYmVsKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgdGhpcy5tZWRpYVN0cmVhbSA9IHRoaXMuY29udGV4dC5jcmVhdGVNZWRpYVN0cmVhbVNvdXJjZShzKTtcclxuICAgICAgICAvLyBzdHJlYW0gY2hhbm5lbCBjb3VudCAoIGlzIGFsd2F5cyAyICEpXHJcbiAgICAgICAgbGV0IHN0cmVhbUNoYW5uZWxDb3VudDogbnVtYmVyID0gdGhpcy5tZWRpYVN0cmVhbS5jaGFubmVsQ291bnQ7XHJcbiAgICAgICAgY29uc29sZS5pbmZvKFwiU3RyZWFtIGNoYW5uZWwgY291bnQ6IFwiK3N0cmVhbUNoYW5uZWxDb3VudCk7XHJcbiAgICAgICAgLy8gaXMgbm90IHNldCEhXHJcbiAgICAgICAgLy90aGlzLmN1cnJlbnRTYW1wbGVSYXRlID0gdGhpcy5tZWRpYVN0cmVhbS5zYW1wbGVSYXRlO1xyXG4gICAgICAgIHRoaXMuY3VycmVudFNhbXBsZVJhdGUgPSB0aGlzLmNvbnRleHQuc2FtcGxlUmF0ZTtcclxuICAgICAgICBjb25zb2xlLmluZm8oXCJTb3VyY2UgYXVkaW8gbm9kZTogY2hhbm5lbHM6IFwiICsgc3RyZWFtQ2hhbm5lbENvdW50ICsgXCIgc2FtcGxlcmF0ZTogXCIgKyB0aGlzLmN1cnJlbnRTYW1wbGVSYXRlKTtcclxuICAgICAgICBpZiAodGhpcy5hdWRpb091dFN0cmVhbSkge1xyXG4gICAgICAgICAgdGhpcy5hdWRpb091dFN0cmVhbS5zZXRGb3JtYXQodGhpcy5jaGFubmVsQ291bnQsIHRoaXMuY3VycmVudFNhbXBsZVJhdGUpO1xyXG4gICAgICAgIH1cclxuICAgICAgICAvLyBXM0MgIC0+IG5ldyBuYW1lIGlzIGNyZWF0ZVNjcmlwdFByb2Nlc3NvclxyXG4gICAgICAgIC8vXHJcbiAgICAgICAgLy8gQWdhaW4gZGVwcmVjYXRlZCwgYnV0IEF1ZGlvV29ya2VyIG5vdCB5ZXQgaW1wbGVtZW50ZWQgaW4gc3RhYmxlIHJlbGVhc2VzIChKdW5lIDIwMTYpXHJcbiAgICAgICAgLy8gQXVkaW9Xb3JrZXIgaXMgbm93IEF1ZGlvV29ya2xldFByb2Nlc3NvciAuLi4gKE1heSAyMDE3KVxyXG5cclxuICAgICAgLy8gVXBkYXRlIDEyLTIwMjA6XHJcbiAgICAgICAvLyBUaGUgU2NyaXB0UHJvY2Vzc29yTm9kZSBJbnRlcmZhY2UgLSBERVBSRUNBVEVEXHJcblxyXG4gICAgICAvLyBVcGRhdGUgMDYtMjAyMVxyXG4gICAgICAvLyAgQXVkaW9Xb3JrbGV0UHJvY2Vzc29yIGlzIGhlcmUgdG8gc3RheS4gV2ViIEF1ZGlvIEFQSSBoYXMgbm93IFJlY29tbWVuZGF0aW9uIHN0YXR1cyAhXHJcblxyXG4gICAgICAgICAgaWYoRU5BQkxFX0FVRElPX1dPUktMRVQgJiYgdGhpcy5jb250ZXh0LmF1ZGlvV29ya2xldCl7XHJcbiAgICAgICAgICAgIC8vY29uc3Qgd29ya2xldEZpbGVOYW1lID0gKCdmaWxlLWxvYWRlciEuL2ludGVyY2VwdG9yX3dvcmtsZXQuanMnKTtcclxuICAgICAgICAgICAgLy9jb25zdCB3b3JrbGV0RmlsZU5hbWUgPSAnaHR0cDovL2xvY2FsaG9zdDo0MjAwL2Fzc2V0cy9pbnRlcmNlcHRvcl93b3JrbGV0LmpzJztcclxuICAgICAgICAgICAgLy9jb25zb2xlLmxvZyhhd3BTdHIpO1xyXG4gICAgICAgICAgICBsZXQgYXVkaW9Xb3JrbGV0TW9kdWxlQmxvYj0gbmV3IEJsb2IoW2F3cFN0cl0sIHt0eXBlOiAndGV4dC9qYXZhc2NyaXB0J30pO1xyXG5cclxuICAgICAgICAgICAgbGV0IGF1ZGlvV29ya2xldE1vZHVsZUJsb2JVcmw9d2luZG93LlVSTC5jcmVhdGVPYmplY3RVUkwoYXVkaW9Xb3JrbGV0TW9kdWxlQmxvYik7XHJcblxyXG4gICAgICAgICAgICB0aGlzLmNvbnRleHQuYXVkaW9Xb3JrbGV0LmFkZE1vZHVsZShhdWRpb1dvcmtsZXRNb2R1bGVCbG9iVXJsKS50aGVuKCgpPT4ge1xyXG4gICAgICAgICAgICAgICAgICBjb25zdCBhd24gPSBuZXcgQXVkaW9Xb3JrbGV0Tm9kZSh0aGlzLmNvbnRleHQsICdjYXB0dXJlLWludGVyY2VwdG9yJyk7XHJcbiAgICAgICAgICAgICAgICAgIGF3bi5vbnByb2Nlc3NvcmVycm9yPShldjpFdmVudCk9PntcclxuICAgICAgICAgICAgICAgICAgICBsZXQgbXNnPSdVbmtud29uIGVycm9yJztcclxuICAgICAgICAgICAgICAgICAgICBpZihldiBpbnN0YW5jZW9mIEVycm9yRXZlbnQpe1xyXG4gICAgICAgICAgICAgICAgICAgICAgbXNnPWV2Lm1lc3NhZ2U7XHJcbiAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IoXCJDYXB0dXJlIGF1ZGlvIHdvcmtsZXQgZXJyb3I6IFwiK21zZyk7XHJcbiAgICAgICAgICAgICAgICAgICAgaWYodGhpcy5saXN0ZW5lcil7XHJcbiAgICAgICAgICAgICAgICAgICAgICB0aGlzLmxpc3RlbmVyLmVycm9yKG1zZyk7XHJcbiAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgIGxldCBhd25QdCA9IGF3bi5wb3J0O1xyXG4gICAgICAgICAgICAgICAgICBpZiAoYXduUHQpIHtcclxuICAgICAgICAgICAgICAgICAgICBhd25QdC5vbm1lc3NhZ2UgPSAoZXY6IE1lc3NhZ2VFdmVudDxhbnk+KSA9PiB7XHJcbiAgICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5jYXB0dXJpbmcpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGR0PWV2LmRhdGE7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIGxldCBjaHMgPSBkdC5jaHM7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIGxldCBhZGFMZW4gPSBkdC5kYXRhLmxlbmd0aDtcclxuICAgICAgICAgICAgICAgICAgICAgICAgaWYoREVCVUdfVFJBQ0VfTEVWRUw+OCkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnNvbGUuZGVidWcoJ1JlY2VpdmVkIGRhdGEgZnJvbSB3b3JrbGV0OiAnICtjaHMgKyAnICcgKyBkdC5sZW4gKycgRGF0YSBjaHM6ICcrYWRhTGVuKTtcclxuICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAvL2xldCBjaHVua0xlbiA9IGFkYUxlbiAvIGNocztcclxuICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGNodW5rTGVuID0gYWRhTGVuO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBsZXQgY2h1bmsgPSBuZXcgQXJyYXk8RmxvYXQzMkFycmF5PihjaHMpO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGxldCBjaCA9IDA7IGNoIDwgY2hzOyBjaCsrKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHRoaXMuZGF0YSAmJiB0aGlzLmRhdGFbY2hdKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXQgYWRhUG9zID0gY2ggKiBjaHVua0xlbjtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmKGR0LmRhdGFbY2hdKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBmYSA9IG5ldyBGbG9hdDMyQXJyYXkoZHQuZGF0YVtjaF0pO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLmRhdGFbY2hdLnB1c2goZmEpO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjaHVua1tjaF0gPSBmYTtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gVXNlIHNhbXBsZXMgb2YgY2hhbm5lbCAwIHRvIGNvdW50IGZyYW1lcyAoc2FtcGxlcylcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGNoID09IDApIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLmZyYW1lc1JlY29yZGVkICs9IGZhLmxlbmd0aDtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfWVsc2V7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmKERFQlVHX1RSQUNFX0xFVkVMPjgpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zb2xlLmRlYnVnKCdDaGFubmVsICcrY2grJyBkYXRhIG5vdCBzZXQhIScpO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0aGlzLmF1ZGlvT3V0U3RyZWFtKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5hdWRpb091dFN0cmVhbS53cml0ZShjaHVuayk7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICB9O1xyXG4gICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgIHRoaXMuYnVmZmVyaW5nTm9kZSA9IGF3bjtcclxuICAgICAgICAgICAgICAgICAgdGhpcy5fb3BlbmVkID0gdHJ1ZTtcclxuICAgICAgICAgICAgICAgICAgaWYgKHRoaXMubGlzdGVuZXIpIHtcclxuICAgICAgICAgICAgICAgICAgICB0aGlzLmxpc3RlbmVyLm9wZW5lZCgpO1xyXG4gICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICkuY2F0Y2goKGVycm9yOiBhbnkpPT57XHJcbiAgICAgICAgICAgICAgY29uc29sZS5sb2coJ0NvdWxkIG5vdCBhZGQgbW9kdWxlICcrZXJyb3IpO1xyXG4gICAgICAgICAgICB9KTtcclxuXHJcbiAgICAgICAgICB9ZWxzZSBpZih0aGlzLmNvbnRleHQuY3JlYXRlU2NyaXB0UHJvY2Vzc29yKSB7XHJcbiAgICAgICAgICAgIC8vIFRoZSBTY3JpcHRQcm9jZXNzb3JOb2RlIEludGVyZmFjZSAtIERFUFJFQ0FURUQgT25seSBhcyBmYWxsYmFja1xyXG4gICAgICAgICAgICAvLyBUT0RPIHNob3VsZCB3ZSB1c2Ugc3RyZWFtQ2hhbm5lbENvdW50IG9yIGNoYW5uZWxDb3VudCBoZXJlID9cclxuICAgICAgICAgICAgbGV0IHNjcmlwdFByb2Nlc3Nvck5vZGU9IHRoaXMuY29udGV4dC5jcmVhdGVTY3JpcHRQcm9jZXNzb3IoQXVkaW9DYXB0dXJlLkJVRkZFUl9TSVpFLCBzdHJlYW1DaGFubmVsQ291bnQsIHN0cmVhbUNoYW5uZWxDb3VudCk7XHJcbiAgICAgICAgICAgIHRoaXMuYnVmZmVyaW5nTm9kZT1zY3JpcHRQcm9jZXNzb3JOb2RlO1xyXG4gICAgICAgICAgICBsZXQgYyA9IDA7XHJcbiAgICAgICAgICAgIGlmKHNjcmlwdFByb2Nlc3Nvck5vZGUub25hdWRpb3Byb2Nlc3Mpe1xyXG4gICAgICAgICAgICAgIHNjcmlwdFByb2Nlc3Nvck5vZGUub25hdWRpb3Byb2Nlc3MgPSAoZTogQXVkaW9Qcm9jZXNzaW5nRXZlbnQpID0+IHtcclxuXHJcbiAgICAgICAgICAgICAgICBpZiAodGhpcy5jYXB0dXJpbmcpIHtcclxuICAgICAgICAgICAgICAgICAgbGV0IGluQnVmZmVyID0gZS5pbnB1dEJ1ZmZlcjtcclxuICAgICAgICAgICAgICAgICAgbGV0IGR1cmF0aW9uID0gaW5CdWZmZXIuZHVyYXRpb247XHJcbiAgICAgICAgICAgICAgICAgIC8vIG9ubHkgcHJvY2VzcyByZXF1ZXN0ZWQgY291bnQgb2YgY2hhbm5lbHNcclxuICAgICAgICAgICAgICAgICAgbGV0IGN1cnJlbnRCdWZmZXJzID0gbmV3IEFycmF5PEZsb2F0MzJBcnJheT4oY2hhbm5lbENvdW50KTtcclxuICAgICAgICAgICAgICAgICAgZm9yIChsZXQgY2g6IG51bWJlciA9IDA7IGNoIDwgY2hhbm5lbENvdW50OyBjaCsrKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgbGV0IGNoU2FtcGxlcyA9IGluQnVmZmVyLmdldENoYW5uZWxEYXRhKGNoKTtcclxuICAgICAgICAgICAgICAgICAgICBsZXQgY2hTYW1wbGVzQ29weSA9IGNoU2FtcGxlcy5zbGljZSgwKTtcclxuICAgICAgICAgICAgICAgICAgICBjdXJyZW50QnVmZmVyc1tjaF0gPSBjaFNhbXBsZXNDb3B5LnNsaWNlKDApO1xyXG4gICAgICAgICAgICAgICAgICAgIHRoaXMuZGF0YVtjaF0ucHVzaChjaFNhbXBsZXNDb3B5KTtcclxuICAgICAgICAgICAgICAgICAgICBpZihERUJVR19UUkFDRV9MRVZFTD44KXtcclxuICAgICAgICAgICAgICAgICAgICAgIGNvbnNvbGUuZGVidWcoXCJQcm9jZXNzIFwiK2NoU2FtcGxlc0NvcHkubGVuZ3RoK1wiIHNhbXBsZXMuXCIpO1xyXG4gICAgICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgICAgICB0aGlzLmZyYW1lc1JlY29yZGVkICs9IGNoU2FtcGxlc0NvcHkubGVuZ3RoO1xyXG4gICAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICAgIGMrKztcclxuICAgICAgICAgICAgICAgICAgaWYgKHRoaXMuYXVkaW9PdXRTdHJlYW0pIHtcclxuICAgICAgICAgICAgICAgICAgICB0aGlzLmF1ZGlvT3V0U3RyZWFtLndyaXRlKGN1cnJlbnRCdWZmZXJzKTtcclxuICAgICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgIH07XHJcbiAgICAgICAgICAgICAgdGhpcy5fb3BlbmVkID0gdHJ1ZTtcclxuICAgICAgICAgICAgICBpZiAodGhpcy5saXN0ZW5lcikge1xyXG4gICAgICAgICAgICAgICAgdGhpcy5saXN0ZW5lci5vcGVuZWQoKTtcclxuICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIH1lbHNle1xyXG4gICAgICAgICAgICAgIHRoaXMubGlzdGVuZXIuZXJyb3IoJ0Jyb3dzZXIgZG9lcyBub3Qgc3VwcG9ydCBhdWRpbyBwcm9jZXNzaW5nIChTY3JpcHRQcm9jZXNzb3Iub25hdWRpb3Byb2Nlc3MgbWV0aG9kIG5vdCBmb3VuZCkhJyk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgIH1lbHNle1xyXG4gICAgICAgICAgICB0aGlzLmxpc3RlbmVyLmVycm9yKCdCcm93c2VyIGRvZXMgbm90IHN1cHBvcnQgYXVkaW8gcHJvY2Vzc2luZyAobmVpdGhlciBBdWRpb1dvcmtsZXRQcm9jZXNzb3Igbm9yIFNjcmlwdFByb2Nlc3NvcikhJyk7XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgfSwgKGUpID0+IHtcclxuICAgICAgICAgIGNvbnNvbGUuZXJyb3IoZSArIFwiIEVycm9yIG5hbWU6IFwiICtlLm5hbWUpO1xyXG4gICAgICAgICAgaWYgKHRoaXMubGlzdGVuZXIpIHtcclxuICAgICAgICAgICAgaWYoJ05vdEFsbG93ZWRFcnJvcicgPT09IGUubmFtZSl7XHJcbiAgICAgICAgICAgICAgdGhpcy5saXN0ZW5lci5lcnJvcignTm90IGFsbG93ZWQgdG8gdXNlIHlvdXIgbWljcm9waG9uZS4nLCdQbGVhc2UgbWFrZSBzdXJlIHRoYXQgbWljcm9waG9uZSBhY2Nlc3MgaXMgYWxsb3dlZCBmb3IgdGhpcyB3ZWIgcGFnZSBhbmQgcmVsb2FkIHRoZSBwYWdlLicpO1xyXG4gICAgICAgICAgICB9ZWxzZSBpZignTm90UmVhZGFibGVFcnJvcicgPT09IGUubmFtZSl7XHJcbiAgICAgICAgICAgICAgdGhpcy5saXN0ZW5lci5lcnJvcignQ291bGQgbm90IHJlYWQgZnJvbSB5b3VyIGF1ZGlvIGRldmljZS4nLCdQbGVhc2UgbWFrZSBzdXJlIHlvdXIgYXVkaW8gZGV2aWNlIGlzIHdvcmtpbmcuJyk7XHJcbiAgICAgICAgICAgIH1lbHNlIGlmKCdPdmVyY29uc3RyYWluZWRFcnJvcicgPT09IGUubmFtZSl7XHJcbiAgICAgICAgICAgICAgbGV0IGVNc2c9ZS5tc2c/ZS5tc2c6J092ZXJjb25zdHJhaW5lZCBtZWRpYSBkZXZpY2UgcmVxdWVzdCBlcnJvci4nO1xyXG4gICAgICAgICAgICAgIHRoaXMubGlzdGVuZXIuZXJyb3IoZU1zZyk7XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgdGhpcy5saXN0ZW5lci5lcnJvcigpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgfVxyXG4gICAgKVxyXG4gIH1cclxuXHJcbiAgc3RhcnQoKSB7XHJcblxyXG4gICAgdGhpcy5pbml0RGF0YSgpO1xyXG4gICAgaWYgKHRoaXMuYXVkaW9PdXRTdHJlYW0pIHtcclxuICAgICAgdGhpcy5hdWRpb091dFN0cmVhbS5uZXh0U3RyZWFtKClcclxuICAgIH1cclxuICAgIHRoaXMuY2FwdHVyaW5nID0gdHJ1ZTtcclxuICAgIGlmKHRoaXMuYnVmZmVyaW5nTm9kZSkge1xyXG4gICAgICB0aGlzLm1lZGlhU3RyZWFtLmNvbm5lY3QodGhpcy5idWZmZXJpbmdOb2RlKTtcclxuICAgICAgdGhpcy5idWZmZXJpbmdOb2RlLmNvbm5lY3QodGhpcy5jb250ZXh0LmRlc3RpbmF0aW9uKTtcclxuICAgIH1cclxuICAgIGlmICh0aGlzLmxpc3RlbmVyKSB7XHJcbiAgICAgIHRoaXMubGlzdGVuZXIuc3RhcnRlZCgpO1xyXG4gICAgfVxyXG5cclxuICB9XHJcblxyXG4gIHN0b3AoKSB7XHJcblxyXG4gICAgaWYgKHRoaXMuZGlzY29ubmVjdFN0cmVhbXMgJiYgdGhpcy5idWZmZXJpbmdOb2RlKSB7XHJcbiAgICAgIHRoaXMubWVkaWFTdHJlYW0uZGlzY29ubmVjdCh0aGlzLmJ1ZmZlcmluZ05vZGUpO1xyXG4gICAgICB0aGlzLmJ1ZmZlcmluZ05vZGUuZGlzY29ubmVjdCh0aGlzLmNvbnRleHQuZGVzdGluYXRpb24pO1xyXG4gICAgfVxyXG5cclxuICAgIGlmICh0aGlzLmF1ZGlvT3V0U3RyZWFtKSB7XHJcbiAgICAgIHRoaXMuYXVkaW9PdXRTdHJlYW0uZmx1c2goKTtcclxuICAgIH1cclxuICAgIHRoaXMuY2FwdHVyaW5nID0gZmFsc2U7XHJcbiAgICBpZiAodGhpcy5saXN0ZW5lcikge1xyXG4gICAgICB0aGlzLmxpc3RlbmVyLnN0b3BwZWQoKTtcclxuICAgIH1cclxuICB9XHJcblxyXG5cclxuICBjbG9zZSgpIHtcclxuICAgIHRoaXMubWVkaWFTdHJlYW0uZGlzY29ubmVjdCgpO1xyXG4gICAgaWYgKHRoaXMuc3RyZWFtKSB7XHJcbiAgICAgIC8vdGhpcy5zdHJlYW0uc3RvcCgpO1xyXG4gICAgICAvLydNZWRpYVN0cmVhbS5zdG9wKCknIGlzIGRlcHJlY2F0ZWQgYW5kIHdpbGwgYmUgcmVtb3ZlZCBpbiBNNDcsIGFyb3VuZCBOb3ZlbWJlciAyMDE1LiBQbGVhc2UgdXNlICdNZWRpYVN0cmVhbS5hY3RpdmUnIGluc3RlYWQuXHJcbiAgICAgIC8vdGhpcy5zdHJlYW0uYWN0aXZlPWZhbHNlO1xyXG4gICAgICB2YXIgbXRzID0gdGhpcy5zdHJlYW0uZ2V0VHJhY2tzKCk7XHJcbiAgICAgIGZvciAodmFyIGkgPSAwOyBpIDwgbXRzLmxlbmd0aDsgaSsrKSB7XHJcbiAgICAgICAgbXRzW2ldLnN0b3AoKTtcclxuICAgICAgfVxyXG4gICAgfVxyXG4gICAgdGhpcy5fb3BlbmVkPWZhbHNlO1xyXG4gIH1cclxuXHJcbiAgYXVkaW9CdWZmZXIoKTogQXVkaW9CdWZmZXIge1xyXG4gICAgdmFyIGZyYW1lTGVuOiBudW1iZXIgPSAwO1xyXG4gICAgdmFyIGNoMERhdGEgPSB0aGlzLmRhdGFbMF07XHJcblxyXG4gICAgZm9yICh2YXIgY2gwQ2hrIG9mIGNoMERhdGEpIHtcclxuICAgICAgZnJhbWVMZW4gKz0gY2gwQ2hrLmxlbmd0aDtcclxuICAgIH1cclxuICAgIHZhciBhYiA9IHRoaXMuY29udGV4dC5jcmVhdGVCdWZmZXIodGhpcy5jaGFubmVsQ291bnQsIGZyYW1lTGVuLCB0aGlzLmNvbnRleHQuc2FtcGxlUmF0ZSk7XHJcbiAgICBmb3IgKHZhciBjaCA9IDA7IGNoIDwgdGhpcy5jaGFubmVsQ291bnQ7IGNoKyspIHtcclxuXHJcbiAgICAgIHZhciBjaEQgPSBhYi5nZXRDaGFubmVsRGF0YShjaCk7XHJcbiAgICAgIHZhciBwb3MgPSAwO1xyXG4gICAgICBmb3IgKHZhciBjaENoayBvZiB0aGlzLmRhdGFbY2hdKSB7XHJcbiAgICAgICAgdmFyIGJ1ZkxlbiA9IGNoQ2hrLmxlbmd0aDtcclxuICAgICAgICBjaEQuc2V0KGNoQ2hrLCBwb3MpO1xyXG4gICAgICAgIHBvcyArPSBidWZMZW47XHJcbiAgICAgIH1cclxuICAgIH1cclxuICAgIHJldHVybiBhYjtcclxuICB9XHJcbn1cclxuXHJcbiJdfQ==