ngx-scandoc 1.2.4 → 1.3.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 (54) hide show
  1. package/core/components/components.module.d.ts +10 -7
  2. package/core/components/manual-scan/manual-scan.component.d.ts +61 -0
  3. package/core/components/scan/scan.component.d.ts +21 -24
  4. package/core/components/scan-results/scan-results.component.d.ts +11 -4
  5. package/core/components/webcam/util/webcam.util.d.ts +1 -1
  6. package/core/components/webcam/webcam/webcam.component.d.ts +26 -180
  7. package/core/components/webcam/webcam.module.d.ts +2 -0
  8. package/core/interfaces/config.d.ts +4 -0
  9. package/dialogs/components/prompt-manual/prompt-manual.component.d.ts +12 -0
  10. package/dialogs/components/scan-profile/scan-profile.component.d.ts +6 -1
  11. package/dialogs/components/scan-selfie/scan-selfie.component.d.ts +15 -61
  12. package/dialogs/dialogs.core.provider.d.ts +1 -0
  13. package/dialogs/dialogs.module.d.ts +12 -9
  14. package/esm2020/core/components/camera-switch/camera-switch.component.mjs +6 -4
  15. package/esm2020/core/components/components.module.mjs +32 -10
  16. package/esm2020/core/components/manual-scan/manual-scan.component.mjs +290 -0
  17. package/esm2020/core/components/scan/scan.component.mjs +290 -181
  18. package/esm2020/core/components/scan-results/scan-results.component.mjs +38 -12
  19. package/esm2020/core/components/webcam/util/webcam.util.mjs +7 -5
  20. package/esm2020/core/components/webcam/webcam/webcam.component.mjs +296 -772
  21. package/esm2020/core/components/webcam/webcam.module.mjs +8 -1
  22. package/esm2020/core/interfaces/config.mjs +1 -1
  23. package/esm2020/dialogs/components/confirm/confirm.component.mjs +3 -3
  24. package/esm2020/dialogs/components/prompt-manual/prompt-manual.component.mjs +30 -0
  25. package/esm2020/dialogs/components/scan-profile/scan-profile.component.mjs +36 -17
  26. package/esm2020/dialogs/components/scan-selfie/scan-selfie.component.mjs +41 -336
  27. package/esm2020/dialogs/dialogs.core.provider.mjs +12 -1
  28. package/esm2020/dialogs/dialogs.module.mjs +24 -11
  29. package/esm2020/forms/types/avatar.type.mjs +26 -21
  30. package/esm2020/forms/types/profile.image.type.mjs +51 -41
  31. package/esm2020/forms/types/title.type.mjs +7 -35
  32. package/esm2020/lib/ngx-scandoc.module.mjs +8 -6
  33. package/esm2020/providers/auth.provider.mjs +15 -2
  34. package/esm2020/providers/camera.provider.mjs +37 -4
  35. package/esm2020/providers/interceptor.provider.mjs +2 -2
  36. package/esm2020/providers/layout.provider.mjs +7 -5
  37. package/esm2020/providers/scan.form.mjs +173 -215
  38. package/esm2020/providers/scan.provider.mjs +264 -7
  39. package/esm2020/providers/translation.provider.mjs +18 -23
  40. package/esm2020/public-api.mjs +3 -1
  41. package/fesm2015/ngx-scandoc.mjs +1722 -1988
  42. package/fesm2015/ngx-scandoc.mjs.map +1 -1
  43. package/fesm2020/ngx-scandoc.mjs +1719 -1982
  44. package/fesm2020/ngx-scandoc.mjs.map +1 -1
  45. package/forms/types/avatar.type.d.ts +1 -0
  46. package/forms/types/profile.image.type.d.ts +6 -1
  47. package/lib/ngx-scandoc.module.d.ts +2 -3
  48. package/package.json +1 -1
  49. package/providers/camera.provider.d.ts +17 -1
  50. package/providers/layout.provider.d.ts +3 -1
  51. package/providers/scan.form.d.ts +2 -0
  52. package/providers/scan.provider.d.ts +6 -1
  53. package/public-api.d.ts +2 -0
  54. package/src/assets/i18n/en.json +14 -3
@@ -1,10 +1,6 @@
1
- import { Component, EventEmitter, HostListener, Input, Output, ViewChild, } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, EventEmitter, HostListener, Input, Output, ViewChild, } from '@angular/core';
2
2
  import { WebcamImage } from '../domain/webcam-image';
3
- import { from, Observable, of } from 'rxjs';
4
- import { WebcamUtil } from '../util/webcam.util';
5
- import { Breakpoints, } from '@angular/cdk/layout';
6
- import { switchMap } from 'rxjs/operators';
7
- import { workertext } from '../../../app.worker';
3
+ import { from } from 'rxjs';
8
4
  import { BaseComponent } from '../../base.component';
9
5
  import * as i0 from "@angular/core";
10
6
  import * as i1 from "@angular/cdk/layout";
@@ -19,313 +15,183 @@ export class WebcamComponent extends BaseComponent {
19
15
  this.cd = cd;
20
16
  this.zone = zone;
21
17
  this.cameraProvider = cameraProvider;
22
- /** Defines the max width of the webcam area in px */
23
- this.width = 640;
24
- /** Defines the max height of the webcam area in px */
25
- this.height = 480;
26
- /** Defines base constraints to apply when requesting video track from UserMedia */
27
- this.videoOptions = WebcamComponent.DEFAULT_VIDEO_OPTIONS;
28
- /** Flag to enable/disable camera switch. If enabled, a switch icon will be displayed if multiple cameras were found */
29
- this.allowCameraSwitch = true;
30
- /** Flag to control whether an ImageData object is stored into the WebcamImage object. */
31
- this.captureImageData = false;
32
- /** The image type to use when capturing snapshots */
33
- this.imageType = WebcamComponent.DEFAULT_IMAGE_TYPE;
34
- /** The image quality to use when capturing snapshots (number between 0 and 1) */
35
- this.imageQuality = WebcamComponent.DEFAULT_IMAGE_QUALITY;
36
- /** EventEmitter which fires when an image has been captured */
37
- this.imageCapture = new EventEmitter();
38
- /** Emits a mediaError if webcam cannot be initialized (e.g. missing user permissions) */
39
- this.initError = new EventEmitter();
40
- /** Emits when the webcam video was clicked */
41
- this.imageClick = new EventEmitter();
42
- /** Emits the active deviceId after the active video device was switched */
18
+ this.canvasSize = { width: 0, height: 0 };
19
+ this.showVideo = false;
20
+ this.canvasData = {
21
+ ctx: null,
22
+ ctxO: null,
23
+ };
24
+ this.chunks = [];
43
25
  this.cameraSwitched = new EventEmitter();
44
26
  this.videoReady = new EventEmitter();
45
- this.showVideo = false;
46
- this.destroyed = new EventEmitter();
47
- /** available video devices */
48
- this.availableVideoInputs = [];
49
- /** Indicates whether the video device is ready to be switched */
50
- this.videoInitialized = false;
51
- this.canvasEl = document.createElement('canvas');
52
- /** Index of active video in availableVideoInputs */
53
- this.activeVideoInputIndex = -1;
54
- /** MediaStream object in use for streaming UserMedia data */
55
- this.mediaStream = null;
56
- this.shutdown = false;
57
- this.canStart = true;
58
- this.videoSize = { width: 1920, height: 1080 };
59
- this.canvasSize = { width: 1280, height: 720 };
60
- this.landscape = true;
61
- // if (typeof Worker !== 'undefined') {
62
- // // Create a new
63
- // this.worker = new Worker(
64
- // new URL('./../../../../app/app.worker', import.meta.url)
65
- // );
66
- // this.worker.onmessage = ({ data }) => {
67
- // console.log(`page got message: ${data}`);
68
- // };
69
- // console.log('POST');
70
- // this.worker.postMessage('hello');
71
- // } else {
72
- // // Web workers are not supported in this environment.
73
- // // You should add a fallback so that your program still executes correctly.
74
- // }
75
- }
76
- /**
77
- * If the given Observable emits, an image will be captured and emitted through 'imageCapture' EventEmitter
78
- */
79
- set trigger(trigger) {
80
- if (this.triggerSubscription) {
81
- this.triggerSubscription.unsubscribe();
82
- }
83
- // Subscribe to events from this Observable to take snapshots
84
- this.triggerSubscription = trigger.subscribe((time) => {
85
- this.takeSnapshot(time);
86
- });
27
+ this.imageCapture = new EventEmitter();
28
+ this.initError = new EventEmitter();
87
29
  }
88
30
  onResize() {
31
+ this.cameraProvider.rectPosition.t = 0;
89
32
  // this.videoReady.next(false);
90
- this.resizeStage();
91
- }
92
- /**
93
- * Get MediaTrackConstraints to request streaming the given device
94
- * @param deviceId
95
- * @param baseMediaTrackConstraints base constraints to merge deviceId-constraint into
96
- * @returns
97
- */
98
- static getMediaConstraintsForDevice(deviceId, baseMediaTrackConstraints) {
99
- const result = baseMediaTrackConstraints
100
- ? baseMediaTrackConstraints
101
- : this.DEFAULT_VIDEO_OPTIONS;
102
- if (deviceId) {
103
- result.deviceId = { exact: deviceId };
33
+ if (this.resizeTimeout) {
34
+ clearTimeout(this.resizeTimeout);
104
35
  }
105
- return result;
36
+ this.resizeTimeout = setTimeout(() => {
37
+ this.drawRectangle();
38
+ }, 50);
106
39
  }
107
- /**
108
- * Tries to harvest the deviceId from the given mediaStreamTrack object.
109
- * Browsers populate this object differently; this method tries some different approaches
110
- * to read the id.
111
- * @param mediaStreamTrack
112
- * @returns deviceId if found in the mediaStreamTrack
113
- */
114
- static getDeviceIdFromMediaStreamTrack(mediaStreamTrack) {
115
- if (mediaStreamTrack.getSettings &&
116
- mediaStreamTrack.getSettings() &&
117
- mediaStreamTrack.getSettings().deviceId) {
118
- return mediaStreamTrack.getSettings().deviceId;
119
- }
120
- else if (mediaStreamTrack.getConstraints &&
121
- mediaStreamTrack.getConstraints() &&
122
- mediaStreamTrack.getConstraints().deviceId) {
123
- const deviceIdObj = mediaStreamTrack.getConstraints().deviceId;
124
- return WebcamComponent.getValueFromConstrainDOMString(deviceIdObj);
125
- }
40
+ ngAfterViewInit() {
41
+ console.warn('[video] start');
42
+ // initial load
43
+ this.__subs(this.cameraProvider.getDevices()).subscribe((resp) => {
44
+ this.init();
45
+ });
46
+ // camera switch
47
+ this.__subs(this.cameraProvider.webcamObservable).subscribe((device) => {
48
+ const videoTrackConstraints = this.setConstraints();
49
+ this.stopAllTracks();
50
+ this.showVideo = false;
51
+ this.cd.detectChanges();
52
+ videoTrackConstraints.deviceId = device;
53
+ this.setup(videoTrackConstraints);
54
+ });
55
+ this.__subs(this.trigger).subscribe((time) => this.takeSnapshot(time));
126
56
  }
127
- /**
128
- * Tries to harvest the facingMode from the given mediaStreamTrack object.
129
- * Browsers populate this object differently; this method tries some different approaches
130
- * to read the value.
131
- * @param mediaStreamTrack
132
- * @returns facingMode if found in the mediaStreamTrack
133
- */
134
- static getFacingModeFromMediaStreamTrack(mediaStreamTrack) {
135
- if (mediaStreamTrack) {
136
- if (mediaStreamTrack.getSettings &&
137
- mediaStreamTrack.getSettings() &&
138
- mediaStreamTrack.getSettings().facingMode) {
139
- return mediaStreamTrack.getSettings().facingMode;
140
- }
141
- else if (mediaStreamTrack.getConstraints &&
142
- mediaStreamTrack.getConstraints() &&
143
- mediaStreamTrack.getConstraints().facingMode) {
144
- const facingModeConstraint = mediaStreamTrack.getConstraints().facingMode;
145
- return WebcamComponent.getValueFromConstrainDOMString(facingModeConstraint);
146
- }
147
- }
57
+ get video() {
58
+ return this.videoRef.nativeElement;
148
59
  }
149
- /**
150
- * Determines whether the given mediaStreamTrack claims itself as user facing
151
- * @param mediaStreamTrack
152
- */
153
- static isUserFacing(mediaStreamTrack) {
154
- const facingMode = WebcamComponent.getFacingModeFromMediaStreamTrack(mediaStreamTrack);
155
- return facingMode ? 'user' === facingMode.toLowerCase() : false;
60
+ get isMobile() {
61
+ return this.platform.ANDROID || this.platform.IOS;
156
62
  }
157
- /**
158
- * Extracts the value from the given ConstrainDOMString
159
- * @param constrainDOMString
160
- */
161
- static getValueFromConstrainDOMString(constrainDOMString) {
162
- if (constrainDOMString) {
163
- if (constrainDOMString instanceof String) {
164
- return String(constrainDOMString);
165
- }
166
- else if (Array.isArray(constrainDOMString) &&
167
- Array(constrainDOMString).length > 0) {
168
- return String(constrainDOMString[0]);
169
- }
170
- else if (typeof constrainDOMString === 'object') {
171
- if (constrainDOMString['exact']) {
172
- return String(constrainDOMString['exact']);
63
+ setConstraints() {
64
+ const out = {};
65
+ switch (this.type) {
66
+ default:
67
+ if (this.isMobile) {
68
+ out.video = { height: { ideal: 1600 }, width: { ideal: 1600 } };
173
69
  }
174
- else if (constrainDOMString['ideal']) {
175
- return String(constrainDOMString['ideal']);
70
+ else {
71
+ out.video = { height: { ideal: 1080 }, width: { ideal: 1920 } };
176
72
  }
177
- }
73
+ out.video.facingMode = 'environment';
74
+ break;
75
+ case 'selfie':
76
+ if (this.isMobile) {
77
+ out.video = { height: { ideal: 1600 }, width: { ideal: 1600 } };
78
+ }
79
+ else {
80
+ out.video = { height: { ideal: 1080 }, width: { ideal: 1920 } };
81
+ }
82
+ out.video.facingMode = 'user';
83
+ break;
178
84
  }
179
- return null;
85
+ return out.video;
180
86
  }
181
- resizeStage() {
182
- setTimeout(() => {
183
- this.updateSize();
184
- this.updatecanvasSize();
185
- this.drawRectangle();
186
- this.cd.detectChanges();
187
- this.videoReady.next(true);
188
- }, 10);
189
- }
190
- get canvasHeight() {
191
- const landscape = this.width > this.height;
192
- const aspect = this.videoSize.width / this.videoSize.height;
193
- if (!landscape) {
194
- return this.width * aspect;
195
- }
196
- return this.height;
87
+ get activeTrackSettings() {
88
+ return this.mediaStream.getVideoTracks()[0].getSettings();
197
89
  }
198
- updatecanvasSize() {
199
- // console.log(this.width, this.height);
200
- const landscape = this.width > this.height;
201
- const aspect = this.videoSize.width / this.videoSize.height;
202
- let width = this.width;
203
- let height = this.height;
204
- if (landscape) {
205
- height = width / aspect;
206
- }
207
- else {
208
- height = width / aspect;
209
- }
210
- if (height > this.height) {
211
- height = this.height;
212
- width = this.height * aspect;
90
+ get activeTrackCapabilities() {
91
+ try {
92
+ return this.mediaStream.getVideoTracks()[0].getCapabilities();
213
93
  }
214
- this.canvasSize = { width: Math.round(width), height: Math.round(height) };
215
- console.log(this.canvasSize, this.videoSize);
216
- if (this.isMobile) {
217
- this.video.nativeElement.setAttribute('height', this.canvasSize.height);
218
- this.video.nativeElement.setAttribute('width', this.canvasSize.width);
94
+ catch (e) {
95
+ return {};
219
96
  }
220
- else {
221
- if (this.videoSize.width > this.videoSize.hight) {
222
- this.video.nativeElement.setAttribute('height', this.canvasSize.height);
223
- }
224
- else {
225
- this.video.nativeElement.setAttribute('width', this.canvasSize.width);
226
- }
227
- }
228
- this.drawRectangle();
229
- this.cd.detectChanges();
230
97
  }
231
- setupWorker() {
232
- if (typeof Worker !== 'undefined') {
233
- // Create a new
234
- if (this.type !== 'selfie') {
235
- this.zone.runOutsideAngular(() => {
236
- var blob = new Blob([workertext], { type: 'text/javascript' });
237
- var url = URL.createObjectURL(blob);
238
- this.worker = new Worker(url);
239
- this.worker.onmessage = ({ data }) => {
240
- if (data.base64) {
241
- data.type == 'data';
242
- this.imageHandler.next(data);
243
- }
244
- };
245
- });
246
- }
247
- }
98
+ init() {
99
+ const videoTrackConstraints = this.setConstraints();
100
+ this.setup(videoTrackConstraints);
248
101
  }
249
- ngAfterViewInit() {
250
- this.setupWorker();
251
- if (this.imageHandler) {
252
- this.__subs(this.imageHandler).subscribe((resp) => {
253
- if (resp.type === 'stop') {
254
- this.worker?.postMessage({ type: 'stop' });
255
- }
256
- });
257
- }
258
- this.__subs(this.breakpointObserver
259
- .observe([Breakpoints.HandsetPortrait, Breakpoints.HandsetLandscape]))
260
- .subscribe((state) => {
261
- if (this.platform.IOS || this.platform.ANDROID) {
262
- // this.landscape = state.breakpoints[Breakpoints.HandsetLandscape];
263
- }
264
- const mobile = this.platform.IOS;
265
- if (state.matches && mobile) {
266
- this.updateSize();
267
- this.updatecanvasSize();
268
- this.drawRectangle();
269
- }
270
- });
271
- if (this.switchCamera) {
272
- if (this.switchCameraSubscription) {
273
- this.switchCameraSubscription.unsubscribe();
274
- }
275
- // Subscribe to events from this Observable to switch video device
276
- this.switchCameraSubscription = this.switchCamera.subscribe((value) => {
277
- this.switchToVideoInput(value);
278
- });
279
- }
280
- this.detectAvailableDevices()
281
- .then(() => {
282
- // start video
283
- if (this.canStart) {
284
- this.switchToVideoInput('');
285
- }
286
- })
287
- .catch((err) => {
288
- this.initError.next({ message: err });
289
- // fallback: still try to load webcam, even if device enumeration failed
290
- if (this.canStart) {
291
- this.switchToVideoInput('');
292
- }
102
+ setup(videoTrackConstraints) {
103
+ this.__subs(from(navigator.mediaDevices.getUserMedia({
104
+ video: videoTrackConstraints,
105
+ }))).subscribe((stream) => {
106
+ this.mediaStream = stream;
107
+ this.cameraProvider.deviceId = this.activeTrackSettings.deviceId;
108
+ this.startVideo();
293
109
  });
294
110
  }
295
- ngOnDestroy() {
296
- this.canStart = false;
297
- this.nativeVideoElement.pause();
298
- this.destroyed.emit(true);
299
- this.stopMediaTracks();
300
- this.unsubscribeFromSubscriptions();
301
- this.shutdown = true;
302
- // shut down worker
303
- if (this.imageHandler) {
304
- this.imageHandler.next({ type: 'stop' });
111
+ startVideo() {
112
+ console.log('START VIDEO');
113
+ this.video.srcObject = this.mediaStream;
114
+ this.video.onloadedmetadata = (data) => {
115
+ // console.log(data);
116
+ this.video.play();
117
+ this.showVideo = true;
118
+ this.cd.detectChanges();
119
+ console.log('VIDEO PLAY');
120
+ };
121
+ this.video.onplay = () => {
122
+ this.videoReady.next(true);
123
+ console.log(this.video.videoWidth, this.video.videoHeight);
124
+ this.drawRectangle();
125
+ };
126
+ }
127
+ setCanvas() {
128
+ let _canvas = document.createElement('canvas'); //this.canvasSnapshot.nativeElement;
129
+ const { videoWidth, videoHeight } = this.video;
130
+ const aspect = videoWidth / videoHeight;
131
+ const canvasSmalSize = 384;
132
+ const smallSize = {
133
+ w: 0,
134
+ h: 0,
135
+ };
136
+ if (videoWidth >= videoHeight) {
137
+ smallSize.w = canvasSmalSize;
138
+ smallSize.h = canvasSmalSize / aspect;
305
139
  }
306
- if (this.nativeVideoElement.srcObject) {
307
- this.nativeVideoElement.srcObject
308
- .getTracks()
309
- .forEach((track) => {
310
- track.stop();
311
- });
312
- this.nativeVideoElement.srcObject = null;
140
+ else {
141
+ smallSize.h = canvasSmalSize;
142
+ smallSize.h = canvasSmalSize * aspect;
313
143
  }
314
- this.__destroy();
315
- // this.nativeVideoElement.src = null;
316
- // this.nativeVideoElement.srcObject = null;
144
+ console.log('SZ', smallSize);
145
+ _canvas.width = videoWidth;
146
+ _canvas.height = videoHeight;
147
+ const ctx = _canvas.getContext('2d', {
148
+ alpha: false,
149
+ powerPreference: 'high-performance',
150
+ antialias: false,
151
+ depth: false,
152
+ desynchronized: true,
153
+ });
154
+ const canvas = document.createElement('canvas'); // needs an initial size
155
+ canvas.height = smallSize.h;
156
+ canvas.width = smallSize.w;
157
+ const ctxO = canvas.getContext('2d', {
158
+ alpha: false,
159
+ powerPreference: 'high-performance',
160
+ antialias: false,
161
+ depth: false,
162
+ desynchronized: true,
163
+ });
164
+ this.canvasData.ctx = ctx;
165
+ this.canvasData.ctxO = ctxO;
166
+ this.canvasData.canvas = canvas;
167
+ this.canvasData._canvas = _canvas;
168
+ this.canvasData.size = smallSize;
169
+ // const stream = this.video.captureStream(5);
170
+ // const mediaRecorder = new MediaRecorder(stream);
171
+ // mediaRecorder.start();
172
+ // mediaRecorder.ondataavailable = (e:any) => {
173
+ // console.log(e)
174
+ // this.chunks.push(e.data);
175
+ // var url = window.URL.createObjectURL(e.data);
176
+ // var anchor = document.createElement('a');
177
+ // anchor.download = 'myfile.webm';
178
+ // anchor.href = url;
179
+ // anchor.click();
180
+ // };
317
181
  }
318
182
  takeSelfie() {
319
- const _video = this.nativeVideoElement;
320
- const mimeType = this.imageType
321
- ? this.imageType
322
- : WebcamComponent.DEFAULT_IMAGE_TYPE;
183
+ const mimeType = 'image/jpeg';
184
+ const { videoWidth, offsetWidth, videoHeight } = this.video;
323
185
  let _canvas = document.createElement('canvas'); //this.canvasSnapshot.nativeElement;
324
- const { width, height } = this.videoSize;
325
- // console.log(width, height);
326
- const { padding, top, rWidth, rHeight } = this.snapRectangle;
327
- _canvas.width = rWidth;
328
- _canvas.height = rHeight;
186
+ const { w, h } = this.cameraProvider.rectPosition;
187
+ const aspect = videoWidth / offsetWidth;
188
+ // console.log(videoWidth, offsetWidth, l, t, w, h);
189
+ _canvas.width = w * aspect;
190
+ _canvas.height = h * aspect;
191
+ const dx = videoWidth - w * aspect;
192
+ const dy = videoHeight - h * aspect;
193
+ const x = dx / 2;
194
+ const y = dy / 2;
329
195
  const ctx = _canvas.getContext('2d', {
330
196
  alpha: false,
331
197
  powerPreference: 'high-performance',
@@ -334,58 +200,69 @@ export class WebcamComponent extends BaseComponent {
334
200
  desynchronized: true,
335
201
  });
336
202
  if (ctx) {
337
- ctx.drawImage(_video, padding, top, rWidth, rHeight, 0, 0, rWidth, rHeight);
203
+ ctx.drawImage(this.video, x, y, w * aspect, h * aspect, 0, 0, w * aspect, h * aspect);
338
204
  const imgAsUrl = _canvas.toDataURL(mimeType, 1);
339
205
  this.imageCapture.emit(new WebcamImage(imgAsUrl, mimeType, new ImageData(1, 1), null));
340
206
  }
341
207
  }
342
- /**
343
- * Takes a snapshot of the current webcam's view and emits the image as an event
344
- */
345
208
  takeSnapshot(time) {
346
- if (this.type === 'selfie') {
209
+ if (this.type !== 'document') {
347
210
  this.takeSelfie();
348
211
  return;
349
212
  }
350
- if (this.trackProcessor) {
213
+ console.warn('SNAP', time);
214
+ if (!this.showVideo) {
351
215
  return;
352
216
  }
353
217
  this.zone.runOutsideAngular(() => {
354
218
  const canvasSmalSize = 384;
355
- const _video = this.nativeVideoElement;
356
- const mimeType = this.imageType
357
- ? this.imageType
358
- : WebcamComponent.DEFAULT_IMAGE_TYPE;
359
- let _canvas = document.createElement('canvas'); //this.canvasSnapshot.nativeElement;
360
- const { width, height } = this.videoSize;
361
- _canvas.width = width;
362
- _canvas.height = height;
363
- const ctx = _canvas.getContext('2d', {
364
- alpha: false,
365
- powerPreference: 'high-performance',
366
- antialias: false,
367
- depth: false,
368
- desynchronized: true,
369
- });
370
- const canvas = document.createElement('canvas'); // needs an initial size
371
- canvas.height = canvasSmalSize;
372
- canvas.width = canvasSmalSize;
373
- const ctxO = canvas.getContext('2d', {
374
- alpha: false,
375
- powerPreference: 'high-performance',
376
- antialias: false,
377
- depth: false,
378
- desynchronized: true,
379
- });
219
+ const mimeType = 'image/jpeg';
220
+ // let _canvas: HTMLCanvasElement = document.createElement('canvas'); //this.canvasSnapshot.nativeElement;
221
+ const { videoWidth, videoHeight } = this.video;
222
+ // _canvas.width = videoWidth;
223
+ // _canvas.height = videoHeight;
224
+ // console.log(videoWidth, videoHeight);
225
+ // const ctx: any = _canvas.getContext('2d', {
226
+ // alpha: false,
227
+ // powerPreference: 'high-performance',
228
+ // antialias: false,
229
+ // depth: false,
230
+ // desynchronized: true,
231
+ // });
232
+ // const canvas = document.createElement('canvas'); // needs an initial size
233
+ // canvas.height = canvasSmalSize;
234
+ // canvas.width = canvasSmalSize;
235
+ // const ctxO: any = canvas.getContext('2d', {
236
+ // alpha: false,
237
+ // powerPreference: 'high-performance',
238
+ // antialias: false,
239
+ // depth: false,
240
+ // desynchronized: true,
241
+ // });
242
+ const { ctx, ctxO, canvas, _canvas, size } = this.canvasData;
380
243
  if (ctx && ctxO) {
381
244
  ctx.imageSmoothingEnabled = false;
382
- ctx.drawImage(_video, 0, 0);
383
- ctxO.drawImage(_video, 0, 0, canvasSmalSize, canvasSmalSize);
384
- const imData = ctx.getImageData(0, 0, this.videoSize.width, this.videoSize.height);
245
+ const drawStart = Date.now();
246
+ ctx.clearRect(0, 0, videoWidth, videoHeight);
247
+ ctxO.clearRect(0, 0, size.w, size.h);
248
+ ctx.drawImage(this.video, 0, 0);
249
+ console.time('drawVideo');
250
+ ctxO.drawImage(_canvas, 0, 0, size.w, size.h);
251
+ console.timeEnd('drawVideo');
252
+ console.time('DRAW');
253
+ const imData = ctx.getImageData(0, 0, videoWidth, videoHeight);
254
+ console.timeEnd('DRAW');
255
+ console.time('CTX');
385
256
  const resize = canvas.toDataURL(mimeType, 1);
257
+ console.timeEnd('CTX');
386
258
  const currentTime = new Date().getTime();
387
259
  const diff = currentTime - time;
388
- const delay = diff > 100 ? 0 : 100 - diff;
260
+ // total time for pack one image
261
+ const drawTime = Date.now() - drawStart;
262
+ console.warn('DRAW TIME', drawTime);
263
+ const delayMax = drawTime < 60 ? 100 : 200;
264
+ const delay = diff > delayMax ? 0 : delayMax - diff;
265
+ console.warn(this.chunks);
389
266
  const timeout = setTimeout(() => {
390
267
  this.imageCapture.emit(new WebcamImage('', mimeType, imData, resize));
391
268
  clearTimeout(timeout);
@@ -393,481 +270,128 @@ export class WebcamComponent extends BaseComponent {
393
270
  }
394
271
  });
395
272
  }
396
- update() { }
397
- resizeImage(base64data) {
398
- return new Observable((observer) => {
399
- let canvas = document.createElement('canvas');
400
- const ctx = canvas.getContext('2d');
401
- const max_size = 384;
402
- let image = new Image();
403
- image.src = base64data;
404
- image.onload = () => {
405
- let width = image.width;
406
- let height = image.height;
407
- if (width < height) {
408
- height *= max_size / width;
409
- width = max_size;
410
- }
411
- else {
412
- width *= max_size / height;
413
- height = max_size;
414
- }
415
- canvas.width = width;
416
- canvas.height = height;
417
- // console.log(width, height);
418
- if (ctx) {
419
- // ctx.rotate((90 * Math.PI) / 180);
420
- // ctx.translate(0, -canvas.width);
421
- ctx.drawImage(image, 0, 0, 384, 384);
422
- observer.next(canvas.toDataURL());
423
- ctx.clearRect(0, 0, 1, 1);
424
- }
425
- else {
426
- observer.error({ type: 'generic error' });
427
- }
428
- image = null;
429
- canvas = null;
430
- };
431
- image.onerror = (e) => {
432
- observer.error(e);
433
- };
434
- });
435
- }
436
- /**
437
- * Switches to the next/previous video device
438
- * @param forward
439
- */
440
- rotateVideoInput(forward) {
441
- if (this.availableVideoInputs && this.availableVideoInputs.length > 1) {
442
- const increment = forward
443
- ? 1
444
- : this.availableVideoInputs.length - 1;
445
- const nextInputIndex = (this.activeVideoInputIndex + increment) %
446
- this.availableVideoInputs.length;
447
- this.switchToVideoInput(this.availableVideoInputs[nextInputIndex].deviceId);
448
- }
449
- }
450
- /**
451
- * Switches the camera-view to the specified video device
452
- */
453
- switchToVideoInput(deviceId) {
454
- // if(deviceId){
455
- this.videoInitialized = false;
456
- this.stopMediaTracks();
457
- this.initWebcam(deviceId, this.videoOptions);
458
- // }
459
- }
460
- /**
461
- * Event-handler for video resize event.
462
- * Triggers Angular change detection so that new video dimensions get applied
463
- */
464
- videoResize() {
465
- // here to trigger Angular change detection
466
- }
467
- get videoWidth() {
468
- const videoRatio = this.getVideoAspectRatio();
469
- return Math.min(this.width, this.height * videoRatio);
470
- }
471
- get videoHeight() {
472
- const videoRatio = this.getVideoAspectRatio();
473
- return Math.min(this.height, this.width / videoRatio);
474
- }
475
- get videoStyleClasses() {
476
- let classes = '';
477
- if (this.isMirrorImage()) {
478
- classes += 'mirrored ';
479
- }
480
- return classes.trim();
481
- }
482
- get nativeVideoElement() {
483
- return this.video.nativeElement;
484
- }
485
- // public get smallVideoElement() {
486
- // return this.videoSmall;
487
- // }
488
- /**
489
- * Returns the video aspect ratio of the active video stream
490
- */
491
- getVideoAspectRatio() {
492
- // calculate ratio from video element dimensions if present
493
- const videoElement = this.nativeVideoElement;
494
- if (videoElement.videoWidth &&
495
- videoElement.videoWidth > 0 &&
496
- videoElement.videoHeight &&
497
- videoElement.videoHeight > 0) {
498
- return videoElement.videoWidth / videoElement.videoHeight;
499
- }
500
- // nothing present - calculate ratio based on width/height params
501
- return this.width / this.height;
502
- }
503
- updateSize() {
504
- const track = this.mediaStream?.getTracks()[0];
505
- if (track) {
506
- let desired = {
507
- width: {
508
- ideal: 0,
509
- },
510
- height: {
511
- ideal: 0,
512
- },
513
- // facingMode:'user',
514
- frameRate: { min: 20, ideal: 24, max: 24 },
515
- };
516
- if (typeof track.getCapabilities === 'function') {
517
- const { width, height } = track.getCapabilities();
518
- desired = {
519
- width: {
520
- ideal: this.landscape && this.isMobile ? height?.max : width?.max,
521
- },
522
- height: {
523
- ideal: this.landscape && this.isMobile ? width?.max : height?.max,
524
- },
525
- // facingMode:'user',
526
- frameRate: { min: 20, ideal: 24, max: 24 },
527
- };
528
- }
529
- else {
530
- desired = {
531
- width: {
532
- ideal: 1920,
533
- },
534
- height: {
535
- ideal: 1080,
536
- },
537
- // facingMode:'user',
538
- frameRate: { min: 20, ideal: 30, max: 31 },
539
- };
540
- }
541
- // desired.height.ideal = 1000;
542
- // desired.width.ideal = 1000;
543
- if (desired.height.ideal && desired.height.ideal >= 1600) {
544
- desired.height.ideal = 1600;
545
- desired.width.ideal = 1600;
546
- }
547
- if (this.platform.ANDROID) {
548
- //desired.width.ideal = 2160;
549
- //desired.height.ideal = 2160;
550
- // if (!this.landscape) {
551
- // } else {
552
- // desired.width.ideal = width?.max;
553
- // desired.height.ideal = height?.max;
554
- // }
555
- }
556
- track.applyConstraints(desired);
557
- //track.getSettings().facingMode!=='user';
558
- this.videoSize = {
559
- height: this.landscape
560
- ? track.getSettings().height
561
- : track.getSettings().width,
562
- width: this.landscape
563
- ? track.getSettings().width
564
- : track.getSettings().height,
565
- };
566
- this.cd.detectChanges();
273
+ stopAllTracks() {
274
+ this.showVideo = false;
275
+ if (this.mediaStream && this.mediaStream.getTracks) {
276
+ // getTracks() returns all media tracks (video+audio)
277
+ this.mediaStream.getTracks().forEach((track) => {
278
+ track.stop();
279
+ this.video.srcObject.removeTrack(track);
280
+ this.mediaStream.removeTrack(track);
281
+ });
282
+ this.video.srcObject = null;
283
+ this.video.load();
567
284
  }
568
285
  }
569
- getStreamTrack(stream) {
570
- return stream.getVideoTracks()[0];
571
- }
572
- getTrackSettings() { }
573
- accesVideoTrack(videoTrackConstraints) {
574
- return from(navigator.mediaDevices.getUserMedia(videoTrackConstraints)).pipe(switchMap((stream) => {
575
- // default resolution 1280x720, check max resolution
576
- const track = this.getStreamTrack(stream);
577
- const capabilities = track.getCapabilities();
578
- const { facingMode, height, width } = capabilities;
579
- const desired = {
580
- width: {
581
- min: this.landscape && this.isMobile ? height?.max : width?.max,
582
- },
583
- // height: {
584
- // exact: this.landscape && this.isMobile ? width?.max : height?.max,
585
- // },
586
- // facingMode:'en',
587
- // frameRate: { min: 25, ideal: 30, max: 31 },
588
- };
589
- // get max width
590
- const { width: sWidth, height: sHeight } = track.getSettings();
591
- if (desired.width.min != sWidth) {
592
- return this.accesVideoTrack({ video: desired });
593
- }
594
- //return this.accesVideoTrack({video:desired});
595
- return of(stream);
596
- }));
597
- }
598
286
  drawRectangle() {
599
- const _canvas = this.canvas.nativeElement;
600
- const ctx = _canvas.getContext('2d');
601
- ctx.clearRect(0, 0, _canvas.width, _canvas.height);
602
- // ctx.drawImage(_video, 0, 0, _canvas.width, _canvas.height);
603
- const { padding, top, rWidth, rHeight } = this.cardRectangle;
604
- ctx.strokeStyle = 'red';
605
- ctx.strokeRect(padding, top, rWidth, rHeight);
606
- }
607
- getMaxAvailableResolution() { }
608
- /**
609
- * Init webcam live view
610
- */
611
- initWebcam(deviceId, userVideoTrackConstraints) {
612
- console.log('INIT WEBCAM');
613
- const _video = this.nativeVideoElement;
614
- // const videoStreamer = this.videoStreamer.nativeElement;
615
- // const videoSmall = document.createElement('video');
616
- if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {
617
- // merge deviceId -> userVideoTrackConstraints
618
- const videoTrackConstraints = WebcamComponent.getMediaConstraintsForDevice(deviceId, userVideoTrackConstraints);
619
- // // if (deviceId) {
620
- // this.accesVideoTrack({ video: videoTrackConstraints }).subscribe(
621
- // (stream: MediaStream) => {
622
- // console.log(
623
- // 'VALID SETTINGS',
624
- // stream.getVideoTracks()[0].getSettings()
625
- // );
626
- // }
627
- // );
628
- // }
629
- if (this.platform.ANDROID) {
630
- videoTrackConstraints.width = 1600;
631
- videoTrackConstraints.height = 1600;
287
+ this.zone.run(() => {
288
+ const _canvas = this.canvas.nativeElement;
289
+ let padding = 10;
290
+ if (!this.isMobile && _canvas.width !== _canvas.height) {
291
+ padding = 40;
632
292
  }
633
- const constraints = {
634
- exact: deviceId,
635
- width: { min: 480, ideal: 1280 },
636
- height: { min: 480, ideal: 720 },
637
- // aspectRatio: 3 / 2,
638
- frameRate: { min: 20 },
639
- };
640
- navigator.mediaDevices
641
- .getUserMedia({ video: videoTrackConstraints })
642
- .then((stream) => {
643
- this.mediaStream = stream;
644
- _video.srcObject = stream;
645
- this.updateSize();
646
- this.updatecanvasSize();
647
- _video.play();
648
- _video.onplay = () => {
649
- this.setActiveCamera(stream);
650
- this.drawRectangle();
651
- };
652
- })
653
- .catch((err) => {
654
- console.log(err);
655
- this.initError.next({
656
- message: err.message,
657
- mediaStreamError: err,
658
- });
659
- });
660
- }
661
- else {
662
- this.initError.next({
663
- message: 'Cannot read UserMedia from MediaDevices.',
664
- });
665
- }
666
- }
667
- get isMobile() {
668
- return this.platform.ANDROID || this.platform.IOS;
669
- }
670
- get cardRectangle() {
671
- const _canvas = this.canvas.nativeElement;
672
- const docSize = this.type === 'selfie' ? 1 : 86 / 55;
673
- // console.log(_canvas.width, _canvas.height);
674
- let padding = 10;
675
- let rWidth = _canvas.width - 2 * padding;
676
- let rHeight = rWidth / docSize;
677
- let top = (_canvas.height - rHeight) / 2;
678
- if ((!this.isMobile && _canvas.width !== _canvas.height)) {
679
- padding = 40;
680
- rHeight = _canvas.height - 2 * padding;
681
- rWidth = rHeight * docSize;
682
- top = (_canvas.height - rHeight) / 2;
683
- padding = (_canvas.width - rWidth) / 2;
684
- }
685
- // console.log(padding, top, rWidth, rHeight);
686
- return { padding, top, rWidth, rHeight };
687
- }
688
- get snapRectangle() {
689
- const _canvas = this.canvas.nativeElement;
690
- const ar = this.videoSize.width / _canvas.width;
691
- let { padding, top, rWidth, rHeight } = this.cardRectangle;
692
- padding = padding * ar;
693
- top = top * ar;
694
- rWidth = rWidth * ar;
695
- rHeight = rHeight * ar;
696
- return { padding, top, rWidth, rHeight };
697
- }
698
- setActiveCamera(stream) {
699
- this.videoReady.next(false);
700
- this.showVideo = false;
701
- this.activeVideoSettings = stream.getVideoTracks()[0].getSettings();
702
- const activeDeviceId = WebcamComponent.getDeviceIdFromMediaStreamTrack(stream.getVideoTracks()[0]);
703
- const videoTrack = stream.getTracks()[0];
704
- this.trackSettings = videoTrack.getSettings();
705
- this.cameraProvider.cameraWasSwitched(activeDeviceId);
706
- this.cameraSwitched.next(activeDeviceId);
707
- setTimeout(() => {
708
- this.zone.run(() => {
709
- this.resizeStage();
710
- this.showVideo = true;
711
- this.videoReady.next(true);
712
- if (MediaStreamTrackProcessor) {
713
- this.trackProcessor = new MediaStreamTrackProcessor(videoTrack);
714
- let frameStream = this.trackProcessor.readable;
715
- if (!this.shutdown) {
716
- this.worker?.postMessage({
717
- type: 'start',
718
- frameStream: frameStream,
719
- trackSettings: this.trackSettings,
720
- }, [frameStream]);
721
- }
293
+ const ctx = _canvas.getContext('2d');
294
+ console.log(ctx);
295
+ ctx.clearRect(0, 0, _canvas.width, _canvas.height);
296
+ const { width, height } = _canvas;
297
+ console.log(width, height);
298
+ const { videoWidth, videoHeight } = this.video;
299
+ const docSize = this.type === 'selfie' ? 1 : 86 / 55;
300
+ const aspect = videoWidth / videoHeight;
301
+ let _height;
302
+ let _width;
303
+ if (videoWidth >= videoHeight) {
304
+ _width = width;
305
+ _height = _width / aspect;
306
+ if (_height > height) {
307
+ _height = height;
308
+ _width = aspect * _height;
722
309
  }
723
- });
724
- }, 500);
725
- }
726
- getActiveVideoTrack() {
727
- return this.mediaStream ? this.mediaStream.getVideoTracks()[0] : null;
728
- }
729
- isMirrorImage() {
730
- if (!this.getActiveVideoTrack()) {
731
- return false;
732
- }
733
- // check for explicit mirror override parameter
734
- {
735
- let mirror = 'auto';
736
- if (this.mirrorImage) {
737
- if (typeof this.mirrorImage === 'string') {
738
- mirror = String(this.mirrorImage).toLowerCase();
739
- }
740
- else {
741
- // WebcamMirrorProperties
742
- if (this.mirrorImage.x) {
743
- mirror = this.mirrorImage.x.toLowerCase();
744
- }
310
+ }
311
+ else {
312
+ _height = height;
313
+ _width = aspect * _height;
314
+ if (_width > width) {
315
+ _width = width;
316
+ _height = _width / aspect;
745
317
  }
746
318
  }
747
- switch (mirror) {
748
- case 'always':
749
- return true;
750
- case 'never':
751
- return false;
319
+ let w, h;
320
+ if (_width <= _height) {
321
+ w = _width - 2 * padding;
322
+ h = w / docSize;
752
323
  }
753
- }
754
- // default: enable mirroring if webcam is user facing
755
- return WebcamComponent.isUserFacing(this.getActiveVideoTrack());
756
- }
757
- /**
758
- * Stops all active media tracks.
759
- * This prevents the webcam from being indicated as active,
760
- * even if it is no longer used by this component.
761
- */
762
- stopMediaTracks() {
763
- // this.video.nativeElement.pause();
764
- if (this.video.nativeElement.srcObject) {
765
- this.video.nativeElement.srcObject
766
- .getTracks()
767
- .forEach((track) => {
768
- track.stop();
769
- });
770
- this.video.nativeElement.srcObject = null;
771
- this.video.nativeElement.src = '';
772
- }
773
- if (this.mediaStream && this.mediaStream.getTracks) {
774
- // getTracks() returns all media tracks (video+audio)
775
- this.mediaStream
776
- .getTracks()
777
- .forEach((track) => track.stop());
778
- }
324
+ else {
325
+ h = _height - 2 * padding;
326
+ w = docSize * h;
327
+ }
328
+ h = Math.floor(h);
329
+ w = Math.floor(w);
330
+ console.log(_width, _height);
331
+ const left = Math.floor((width - w) / 2);
332
+ const top = Math.floor((height - h) / 2);
333
+ ctx.strokeStyle = 'red';
334
+ // ctx.strokeRect(left, top, _width, _height);
335
+ ctx.shadowColor = 'red';
336
+ ctx.shadowBlur = 15;
337
+ ctx.beginPath();
338
+ // Draw using 5px for border radius on all sides
339
+ // stroke it but no fill
340
+ ctx.roundRect(left, top, w, h, 5);
341
+ ctx.stroke();
342
+ this.cameraProvider.rectPosition = {
343
+ l: left,
344
+ t: top,
345
+ w,
346
+ h,
347
+ _w: _width,
348
+ _h: _height,
349
+ };
350
+ this.setCanvas();
351
+ // ctx.stroke();
352
+ });
779
353
  }
780
- /**
781
- * Unsubscribe from all open subscriptions
782
- */
783
- unsubscribeFromSubscriptions() {
354
+ ngOnDestroy() {
355
+ this.video.pause();
356
+ // this.video.getTracks().forEach((track: any) => {
357
+ // track.stop();
358
+ // this.video.srcObject.removeTrack(track);
359
+ // });
360
+ this.stopAllTracks();
784
361
  if (this.triggerSubscription) {
785
362
  this.triggerSubscription.unsubscribe();
786
363
  }
787
- if (this.switchCameraSubscription) {
788
- this.switchCameraSubscription.unsubscribe();
789
- }
790
- }
791
- /**
792
- * Reads available input devices
793
- */
794
- detectAvailableDevices() {
795
- return new Promise((resolve, reject) => {
796
- WebcamUtil.getAvailableVideoInputs().subscribe((devices) => {
797
- this.availableVideoInputs = devices;
798
- resolve(devices);
799
- }, (err) => {
800
- this.availableVideoInputs = [];
801
- reject(err);
802
- });
803
- });
364
+ this.__destroy();
365
+ console.warn('[VIDEO] destroyed');
804
366
  }
805
367
  }
806
- WebcamComponent.DEFAULT_VIDEO_OPTIONS = {
807
- facingMode: 'environment',
808
- };
809
- WebcamComponent.DEFAULT_IMAGE_TYPE = 'image/jpeg';
810
- WebcamComponent.DEFAULT_IMAGE_QUALITY = 1;
811
368
  WebcamComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: WebcamComponent, deps: [{ token: i1.BreakpointObserver }, { token: i2.Platform }, { token: i0.ChangeDetectorRef }, { token: i0.NgZone }, { token: i3.NgxScandocCameraProvider }], target: i0.ɵɵFactoryTarget.Component });
812
- WebcamComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: WebcamComponent, selector: "ngx-scandoc-webcam", inputs: { imageHandler: "imageHandler", id: "id", type: "type", width: "width", height: "height", videoOptions: "videoOptions", allowCameraSwitch: "allowCameraSwitch", mirrorImage: "mirrorImage", captureImageData: "captureImageData", imageType: "imageType", imageQuality: "imageQuality", trigger: "trigger", switchCamera: "switchCamera" }, outputs: { imageCapture: "imageCapture", initError: "initError", imageClick: "imageClick", cameraSwitched: "cameraSwitched", videoReady: "videoReady", destroyed: "destroyed" }, host: { listeners: { "window:resize": "onResize($event)" } }, viewQueries: [{ propertyName: "video", first: true, predicate: ["video"], descendants: true, static: true }, { propertyName: "videoStreamer", first: true, predicate: ["videoStreamer"], descendants: true, static: true }, { propertyName: "canvas", first: true, predicate: ["canvas"], descendants: true, static: true }, { propertyName: "canvasSnapshot", first: true, predicate: ["canvasSnapshot"], descendants: true, static: true }, { propertyName: "canvasResize", first: true, predicate: ["canvasResize"], descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<div class=\"webcam-wrapper\">\n <div>\n <video\n #video\n autoplay\n muted\n style=\"display: block\"\n playsinline\n\n ></video>\n\n <div class=\"rectangle\">\n\n <canvas\n #canvas\n [ngStyle]=\"{ visibility: showVideo ? 'visible' : 'hidden' }\"\n [width]=\"canvasSize.width\"\n [height]=\"canvasSize.height\"\n ></canvas>\n </div>\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:row;flex:1;transform:transale3d(0,0,0);background-color:#000}.webcam-wrapper{display:flex;flex-direction:row;align-items:center;flex:1;justify-content:center;position:relative}.webcam-wrapper canvas{transform:transale3d(0,0,0);display:block;position:relative;z-index:20;left:0;top:0}.webcam-wrapper .rectangle{position:absolute;left:0;top:0;display:flex;flex-direction:row;align-items:center;justify-content:center;width:100%;height:100%}\n"], dependencies: [{ kind: "directive", type: i4.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
369
+ WebcamComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: WebcamComponent, selector: "ngx-scandoc-webcam", inputs: { imageHandler: "imageHandler", type: "type", trigger: "trigger" }, outputs: { cameraSwitched: "cameraSwitched", videoReady: "videoReady", imageCapture: "imageCapture", initError: "initError" }, host: { listeners: { "window:resize": "onResize($event)" } }, viewQueries: [{ propertyName: "canvas", first: true, predicate: ["canvas"], descendants: true, static: true }, { propertyName: "videoRef", first: true, predicate: ["video"], descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<div class=\"webcam-wrapper\">\n <video #video autoplay muted style=\"display: block\" playsinline></video>\n\n <div class=\"rectangle\">\n <canvas\n #canvas\n [ngStyle]=\"{ visibility: showVideo ? 'visible' : 'hidden' }\"\n [width]=\"video.offsetWidth\"\n [height]=\"video.offsetHeight\"\n ></canvas>\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:row;flex:1;transform:transale3d(0,0,0);background-color:#000;overflow:hidden}.webcam-wrapper{display:flex;overflow:hidden;flex-direction:row;flex:1;position:relative}.webcam-wrapper video{overflow:hidden;flex:1;object-fit:contain}.webcam-wrapper canvas{transform:translateZ(0);display:block;position:absolute;z-index:20;left:0;top:0}.webcam-wrapper .rectangle{position:absolute;left:0;top:0;display:flex;flex-direction:row;align-items:center;justify-content:center;width:100%;height:100%}\n"], dependencies: [{ kind: "directive", type: i4.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
813
370
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: WebcamComponent, decorators: [{
814
371
  type: Component,
815
- args: [{ selector: 'ngx-scandoc-webcam', template: "<div class=\"webcam-wrapper\">\n <div>\n <video\n #video\n autoplay\n muted\n style=\"display: block\"\n playsinline\n\n ></video>\n\n <div class=\"rectangle\">\n\n <canvas\n #canvas\n [ngStyle]=\"{ visibility: showVideo ? 'visible' : 'hidden' }\"\n [width]=\"canvasSize.width\"\n [height]=\"canvasSize.height\"\n ></canvas>\n </div>\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:row;flex:1;transform:transale3d(0,0,0);background-color:#000}.webcam-wrapper{display:flex;flex-direction:row;align-items:center;flex:1;justify-content:center;position:relative}.webcam-wrapper canvas{transform:transale3d(0,0,0);display:block;position:relative;z-index:20;left:0;top:0}.webcam-wrapper .rectangle{position:absolute;left:0;top:0;display:flex;flex-direction:row;align-items:center;justify-content:center;width:100%;height:100%}\n"] }]
372
+ args: [{ selector: 'ngx-scandoc-webcam', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"webcam-wrapper\">\n <video #video autoplay muted style=\"display: block\" playsinline></video>\n\n <div class=\"rectangle\">\n <canvas\n #canvas\n [ngStyle]=\"{ visibility: showVideo ? 'visible' : 'hidden' }\"\n [width]=\"video.offsetWidth\"\n [height]=\"video.offsetHeight\"\n ></canvas>\n </div>\n</div>\n", styles: [":host{display:flex;flex-direction:row;flex:1;transform:transale3d(0,0,0);background-color:#000;overflow:hidden}.webcam-wrapper{display:flex;overflow:hidden;flex-direction:row;flex:1;position:relative}.webcam-wrapper video{overflow:hidden;flex:1;object-fit:contain}.webcam-wrapper canvas{transform:translateZ(0);display:block;position:absolute;z-index:20;left:0;top:0}.webcam-wrapper .rectangle{position:absolute;left:0;top:0;display:flex;flex-direction:row;align-items:center;justify-content:center;width:100%;height:100%}\n"] }]
816
373
  }], ctorParameters: function () { return [{ type: i1.BreakpointObserver }, { type: i2.Platform }, { type: i0.ChangeDetectorRef }, { type: i0.NgZone }, { type: i3.NgxScandocCameraProvider }]; }, propDecorators: { imageHandler: [{
817
374
  type: Input
818
- }], id: [{
819
- type: Input
820
375
  }], type: [{
821
376
  type: Input
822
- }], width: [{
823
- type: Input
824
- }], height: [{
825
- type: Input
826
- }], videoOptions: [{
827
- type: Input
828
- }], allowCameraSwitch: [{
829
- type: Input
830
- }], mirrorImage: [{
831
- type: Input
832
- }], captureImageData: [{
833
- type: Input
834
- }], imageType: [{
835
- type: Input
836
- }], imageQuality: [{
377
+ }], trigger: [{
837
378
  type: Input
838
- }], imageCapture: [{
839
- type: Output
840
- }], initError: [{
841
- type: Output
842
- }], imageClick: [{
843
- type: Output
379
+ }], onResize: [{
380
+ type: HostListener,
381
+ args: ['window:resize', ['$event']]
382
+ }], canvas: [{
383
+ type: ViewChild,
384
+ args: ['canvas', { static: true }]
385
+ }], videoRef: [{
386
+ type: ViewChild,
387
+ args: ['video', { static: true }]
844
388
  }], cameraSwitched: [{
845
389
  type: Output
846
390
  }], videoReady: [{
847
391
  type: Output
848
- }], destroyed: [{
392
+ }], imageCapture: [{
393
+ type: Output
394
+ }], initError: [{
849
395
  type: Output
850
- }], video: [{
851
- type: ViewChild,
852
- args: ['video', { static: true }]
853
- }], videoStreamer: [{
854
- type: ViewChild,
855
- args: ['videoStreamer', { static: true }]
856
- }], canvas: [{
857
- type: ViewChild,
858
- args: ['canvas', { static: true }]
859
- }], canvasSnapshot: [{
860
- type: ViewChild,
861
- args: ['canvasSnapshot', { static: true }]
862
- }], canvasResize: [{
863
- type: ViewChild,
864
- args: ['canvasResize', { static: true }]
865
- }], trigger: [{
866
- type: Input
867
- }], switchCamera: [{
868
- type: Input
869
- }], onResize: [{
870
- type: HostListener,
871
- args: ['window:resize', ['$event']]
872
396
  }] } });
873
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2ViY2FtLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC1zY2FuZG9jL3NyYy9jb3JlL2NvbXBvbmVudHMvd2ViY2FtL3dlYmNhbS93ZWJjYW0uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LXNjYW5kb2Mvc3JjL2NvcmUvY29tcG9uZW50cy93ZWJjYW0vd2ViY2FtL3dlYmNhbS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBR0wsU0FBUyxFQUNULFlBQVksRUFDWixZQUFZLEVBQ1osS0FBSyxFQUdMLE1BQU0sRUFDTixTQUFTLEdBQ1YsTUFBTSxlQUFlLENBQUM7QUFFdkIsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQ3JELE9BQU8sRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFLEVBQUUsRUFBeUIsTUFBTSxNQUFNLENBQUM7QUFDbkUsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBRWpELE9BQU8sRUFFTCxXQUFXLEdBRVosTUFBTSxxQkFBcUIsQ0FBQztBQUU3QixPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDM0MsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBRWpELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQzs7Ozs7O0FBU3JELE1BQU0sT0FBTyxlQUNYLFNBQVEsYUFBYTtJQXdIckIsWUFDUyxrQkFBc0MsRUFDdEMsUUFBa0IsRUFDakIsRUFBcUIsRUFDckIsSUFBWSxFQUVaLGNBQXdDO1FBRWhELEtBQUssRUFBRSxDQUFDO1FBUEQsdUJBQWtCLEdBQWxCLGtCQUFrQixDQUFvQjtRQUN0QyxhQUFRLEdBQVIsUUFBUSxDQUFVO1FBQ2pCLE9BQUUsR0FBRixFQUFF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ZUFBZSxFQUFFLEVBQUUsTUFBTSxFQUFFLElBQUksRUFBRTtnQkFLRyxNQUFNO3NCQUFwRCxTQUFTO3VCQUFDLFFBQVEsRUFBRSxFQUFFLE1BQU0sRUFBRSxJQUFJLEVBQUU7Z0JBR2tCLGNBQWM7c0JBQXBFLFNBQVM7dUJBQUMsZ0JBQWdCLEVBQUUsRUFBRSxNQUFNLEVBQUUsSUFBSSxFQUFFO2dCQUVRLFlBQVk7c0JBQWhFLFNBQVM7dUJBQUMsY0FBYyxFQUFFLEVBQUUsTUFBTSxFQUFFLElBQUksRUFBRTtnQkFjaEMsT0FBTztzQkFEakIsS0FBSztnQkFtQkcsWUFBWTtzQkFBcEIsS0FBSztnQkFHTixRQUFRO3NCQURQLFlBQVk7dUJBQUMsZUFBZSxFQUFFLENBQUMsUUFBUSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQWZ0ZXJWaWV3SW5pdCxcbiAgQ2hhbmdlRGV0ZWN0b3JSZWYsXG4gIENvbXBvbmVudCxcbiAgRXZlbnRFbWl0dGVyLFxuICBIb3N0TGlzdGVuZXIsXG4gIElucHV0LFxuICBOZ1pvbmUsXG4gIE9uRGVzdHJveSxcbiAgT3V0cHV0LFxuICBWaWV3Q2hpbGQsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgV2ViY2FtSW5pdEVycm9yIH0gZnJvbSAnLi4vZG9tYWluL3dlYmNhbS1pbml0LWVycm9yJztcbmltcG9ydCB7IFdlYmNhbUltYWdlIH0gZnJvbSAnLi4vZG9tYWluL3dlYmNhbS1pbWFnZSc7XG5pbXBvcnQgeyBmcm9tLCBPYnNlcnZhYmxlLCBvZiwgU3ViamVjdCwgU3Vic2NyaXB0aW9uIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBXZWJjYW1VdGlsIH0gZnJvbSAnLi4vdXRpbC93ZWJjYW0udXRpbCc7XG5pbXBvcnQgeyBXZWJjYW1NaXJyb3JQcm9wZXJ0aWVzIH0gZnJvbSAnLi4vZG9tYWluL3dlYmNhbS1taXJyb3ItcHJvcGVydGllcyc7XG5pbXBvcnQge1xuICBCcmVha3BvaW50T2JzZXJ2ZXIsXG4gIEJyZWFrcG9pbnRzLFxuICBCcmVha3BvaW50U3RhdGUsXG59IGZyb20gJ0Bhbmd1bGFyL2Nkay9sYXlvdXQnO1xuaW1wb3J0IHsgUGxhdGZvcm0gfSBmcm9tICdAYW5ndWxhci9jZGsvcGxhdGZvcm0nO1xuaW1wb3J0IHsgc3dpdGNoTWFwIH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHsgd29ya2VydGV4dCB9IGZyb20gJy4uLy4uLy4uL2FwcC53b3JrZXInO1xuaW1wb3J0IHsgTmd4U2NhbmRvY0NhbWVyYVByb3ZpZGVyIH0gZnJvbSAnLi4vLi4vLi4vLi4vcHJvdmlkZXJzL2NhbWVyYS5wcm92aWRlcic7XG5pbXBvcnQgeyBCYXNlQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYmFzZS5jb21wb25lbnQnO1xuXG5kZWNsYXJlIHZhciBNZWRpYVN0cmVhbVRyYWNrUHJvY2Vzc29yOiBhbnk7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ25neC1zY2FuZG9jLXdlYmNhbScsXG4gIHRlbXBsYXRlVXJsOiAnLi93ZWJjYW0uY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi93ZWJjYW0uY29tcG9uZW50LnNjc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgV2ViY2FtQ29tcG9uZW50XG4gIGV4dGVuZHMgQmFzZUNvbXBvbmVudFxuICBpbXBsZW1lbnRzIEFmdGVyVmlld0luaXQsIE9uRGVzdHJveVxue1xuICB0cmFja1Byb2Nlc3NvcjogYW55O1xuICB0cmFja1NldHRpbmdzOiBhbnk7XG5cbiAgQElucHV0KCkgaW1hZ2VIYW5kbGVyITogU3ViamVjdDxhbnk+O1xuXG4gIHByaXZhdGUgc3RhdGljIERFRkFVTFRfVklERU9fT1BUSU9OUzogTWVkaWFUcmFja0NvbnN0cmFpbnRzID0ge1xuICAgIGZhY2luZ01vZGU6ICdlbnZpcm9ubWVudCcsXG4gIH07XG4gIHByaXZhdGUgc3RhdGljIERFRkFVTFRfSU1BR0VfVFlQRTogc3RyaW5nID0gJ2ltYWdlL2pwZWcnO1xuICBwcml2YXRlIHN0YXRpYyBERUZBVUxUX0lNQUdFX1FVQUxJVFk6IG51bWJlciA9IDE7XG5cbiAgQElucHV0KCkgcHVibGljIGlkPzogYW55O1xuICBASW5wdXQoKSBwdWJsaWMgdHlwZT86ICdkb2N1bWVudCcgfCAnc2VsZmllJztcbiAgLyoqIERlZmluZXMgdGhlIG1heCB3aWR0aCBvZiB0aGUgd2ViY2FtIGFyZWEgaW4gcHggKi9cbiAgQElucHV0KCkgcHVibGljIHdpZHRoOiBudW1iZXIgPSA2NDA7XG4gIC8qKiBEZWZpbmVzIHRoZSBtYXggaGVpZ2h0IG9mIHRoZSB3ZWJjYW0gYXJlYSBpbiBweCAqL1xuICBASW5wdXQoKSBwdWJsaWMgaGVpZ2h0OiBudW1iZXIgPSA0ODA7XG4gIC8qKiBEZWZpbmVzIGJhc2UgY29uc3RyYWludHMgdG8gYXBwbHkgd2hlbiByZXF1ZXN0aW5nIHZpZGVvIHRyYWNrIGZyb20gVXNlck1lZGlhICovXG4gIEBJbnB1dCgpIHB1YmxpYyB2aWRlb09wdGlvbnM6IE1lZGlhVHJhY2tDb25zdHJhaW50cyA9XG4gICAgV2ViY2FtQ29tcG9uZW50LkRFRkFVTFRfVklERU9fT1BUSU9OUztcbiAgLyoqIEZsYWcgdG8gZW5hYmxlL2Rpc2FibGUgY2FtZXJhIHN3aXRjaC4gSWYgZW5hYmxlZCwgYSBzd2l0Y2ggaWNvbiB3aWxsIGJlIGRpc3BsYXllZCBpZiBtdWx0aXBsZSBjYW1lcmFzIHdlcmUgZm91bmQgKi9cbiAgQElucHV0KCkgcHVibGljIGFsbG93Q2FtZXJhU3dpdGNoOiBib29sZWFuID0gdHJ1ZTtcbiAgLyoqIFBhcmFtZXRlciB0byBjb250cm9sIGltYWdlIG1pcnJvcmluZyAoaS5lLiBmb3IgdXNlci1mYWNpbmcgY2FtZXJhKS4gW1wiYXV0b1wiLCBcImFsd2F5c1wiLCBcIm5ldmVyXCJdICovXG4gIEBJbnB1dCgpIHB1YmxpYyBtaXJyb3JJbWFnZT86IHN0cmluZyB8IFdlYmNhbU1pcnJvclByb3BlcnRpZXM7XG4gIC8qKiBGbGFnIHRvIGNvbnRyb2wgd2hldGhlciBhbiBJbWFnZURhdGEgb2JqZWN0IGlzIHN0b3JlZCBpbnRvIHRoZSBXZWJjYW1JbWFnZSBvYmplY3QuICovXG4gIEBJbnB1dCgpIHB1YmxpYyBjYXB0dXJlSW1hZ2VEYXRhOiBib29sZWFuID0gZmFsc2U7XG4gIC8qKiBUaGUgaW1hZ2UgdHlwZSB0byB1c2Ugd2hlbiBjYXB0dXJpbmcgc25hcHNob3RzICovXG4gIEBJbnB1dCgpIHB1YmxpYyBpbWFnZVR5cGU6IHN0cmluZyA9IFdlYmNhbUNvbXBvbmVudC5ERUZBVUxUX0lNQUdFX1RZUEU7XG4gIC8qKiBUaGUgaW1hZ2UgcXVhbGl0eSB0byB1c2Ugd2hlbiBjYXB0dXJpbmcgc25hcHNob3RzIChudW1iZXIgYmV0d2VlbiAwIGFuZCAxKSAqL1xuICBASW5wdXQoKSBwdWJsaWMgaW1hZ2VRdWFsaXR5OiBudW1iZXIgPSBXZWJjYW1Db21wb25lbnQuREVGQVVMVF9JTUFHRV9RVUFMSVRZO1xuXG4gIC8qKiBFdmVudEVtaXR0ZXIgd2hpY2ggZmlyZXMgd2hlbiBhbiBpbWFnZSBoYXMgYmVlbiBjYXB0dXJlZCAqL1xuICBAT3V0cHV0KCkgcHVibGljIGltYWdlQ2FwdHVyZTogRXZlbnRFbWl0dGVyPFdlYmNhbUltYWdlPiA9XG4gICAgbmV3IEV2ZW50RW1pdHRlcjxXZWJjYW1JbWFnZT4oKTtcbiAgLyoqIEVtaXRzIGEgbWVkaWFFcnJvciBpZiB3ZWJjYW0gY2Fubm90IGJlIGluaXRpYWxpemVkIChlLmcuIG1pc3NpbmcgdXNlciBwZXJtaXNzaW9ucykgKi9cbiAgQE91dHB1dCgpIHB1YmxpYyBpbml0RXJyb3I6IEV2ZW50RW1pdHRlcjxXZWJjYW1Jbml0RXJyb3I+ID1cbiAgICBuZXcgRXZlbnRFbWl0dGVyPFdlYmNhbUluaXRFcnJvcj4oKTtcbiAgLyoqIEVtaXRzIHdoZW4gdGhlIHdlYmNhbSB2aWRlbyB3YXMgY2xpY2tlZCAqL1xuICBAT3V0cHV0KCkgcHVibGljIGltYWdlQ2xpY2s6IEV2ZW50RW1pdHRlcjx2b2lkPiA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcbiAgLyoqIEVtaXRzIHRoZSBhY3RpdmUgZGV2aWNlSWQgYWZ0ZXIgdGhlIGFjdGl2ZSB2aWRlbyBkZXZpY2Ugd2FzIHN3aXRjaGVkICovXG4gIEBPdXRwdXQoKSBwdWJsaWMgY2FtZXJhU3dpdGNoZWQ6IEV2ZW50RW1pdHRlcjxzdHJpbmc+ID1cbiAgICBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZz4oKTtcblxuICBAT3V0cHV0KCkgcHVibGljIHZpZGVvUmVhZHk6IEV2ZW50RW1pdHRlcjxib29sZWFuPiA9XG4gICAgbmV3IEV2ZW50RW1pdHRlcjxib29sZWFuPigpO1xuICBzaG93VmlkZW8gPSBmYWxzZTtcbiAgQE91dHB1dCgpIGRlc3Ryb3llZCA9IG5ldyBFdmVudEVtaXR0ZXIoKTtcblxuICAvKiogYXZhaWxhYmxlIHZpZGVvIGRldmljZXMgKi9cbiAgcHVibGljIGF2YWlsYWJsZVZpZGVvSW5wdXRzOiBNZWRpYURldmljZUluZm9bXSA9IFtdO1xuXG4gIC8qKiBJbmRpY2F0ZXMgd2hldGhlciB0aGUgdmlkZW8gZGV2aWNlIGlzIHJlYWR5IHRvIGJlIHN3aXRjaGVkICovXG4gIHB1YmxpYyB2aWRlb0luaXRpYWxpemVkOiBib29sZWFuID0gZmFsc2U7XG5cbiAgY2FudmFzRWwgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdjYW52YXMnKTtcblxuICAvKiogSWYgdGhlIE9ic2VydmFibGUgcmVwcmVzZW50ZWQgYnkgdGhpcyBzdWJzY3JpcHRpb24gZW1pdHMsIGFuIGltYWdlIHdpbGwgYmUgY2FwdHVyZWQgYW5kIGVtaXR0ZWQgdGhyb3VnaFxuICAgKiB0aGUgJ2ltYWdlQ2FwdHVyZScgRXZlbnRFbWl0dGVyICovXG4gIHByaXZhdGUgdHJpZ2dlclN1YnNjcmlwdGlvbj86IFN1YnNjcmlwdGlvbjtcbiAgLyoqIEluZGV4IG9mIGFjdGl2ZSB2aWRlbyBpbiBhdmFpbGFibGVWaWRlb0lucHV0cyAqL1xuICBwcml2YXRlIGFjdGl2ZVZpZGVvSW5wdXRJbmRleDogbnVtYmVyID0gLTE7XG4gIC8qKiBTdWJzY3JpcHRpb24gdG8gc3dpdGNoQ2FtZXJhIGV2ZW50cyAqL1xuICBwcml2YXRlIHN3aXRjaENhbWVyYVN1YnNjcmlwdGlvbj86IFN1YnNjcmlwdGlvbjtcbiAgLyoqIE1lZGlhU3RyZWFtIG9iamVjdCBpbiB1c2UgZm9yIHN0cmVhbWluZyBVc2VyTWVkaWEgZGF0YSAqL1xuICBwcml2YXRlIG1lZGlhU3RyZWFtOiBNZWRpYVN0cmVhbSB8IG51bGwgPSBudWxsO1xuICAvLyBPcmlnaW5hbCB2aWRlb1xuICBAVmlld0NoaWxkKCd2aWRlbycsIHsgc3RhdGljOiB0cnVlIH0pIHByaXZhdGUgdmlkZW86IGFueTtcbiAgLy8gQWRqdXN0ZWQgZm9yIFVJXG4gIEBWaWV3Q2hpbGQoJ3ZpZGVvU3RyZWFtZXInLCB7IHN0YXRpYzogdHJ1ZSB9KSBwcml2YXRlIHZpZGVvU3RyZWFtZXI6IGFueTtcblxuICAvL0BWaWV3Q2hpbGQoJ3ZpZGVvU21hbGwnLCB7IHN0YXRpYzogdHJ1ZSB9KSBwcml2YXRlIHZpZGVvU21hbGw6IGFueTtcbiAgLy92aWRlb1NtYWxsOiBIVE1MVmlkZW9FbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgndmlkZW8nKTtcbiAgLyoqIENhbnZhcyBmb3IgVmlkZW8gU25hcHNob3RzICovXG4gIEBWaWV3Q2hpbGQoJ2NhbnZhcycsIHsgc3RhdGljOiB0cnVlIH0pIHByaXZhdGUgY2FudmFzOiBhbnk7XG5cbiAgLyoqIENhbnZhcyBmb3IgVmlkZW8gU25hcHNob3RzICovXG4gIEBWaWV3Q2hpbGQoJ2NhbnZhc1NuYXBzaG90JywgeyBzdGF0aWM6IHRydWUgfSkgcHJpdmF0ZSBjYW52YXNTbmFwc2hvdDogYW55O1xuXG4gIEBWaWV3Q2hpbGQoJ2NhbnZhc1Jlc2l6ZScsIHsgc3RhdGljOiB0cnVlIH0pIHByaXZhdGUgY2FudmFzUmVzaXplOiBhbnk7XG4gIC8qKiB3aWR0aCBhbmQgaGVpZ2h0IG9mIHRoZSBhY3RpdmUgdmlkZW8gc3RyZWFtICovXG4gIGFjdGl2ZVZpZGVvU2V0dGluZ3M6IGFueTtcblxuICBzaHV0ZG93biA9IGZhbHNlO1xuICBwcml2YXRlIGNhblN0YXJ0ID0gdHJ1ZTtcbiAgdmlkZW9TaXplOiBhbnkgPSB7IHdpZHRoOiAxOTIwLCBoZWlnaHQ6IDEwODAgfTtcbiAgY2FudmFzU2l6ZTogYW55ID0geyB3aWR0aDogMTI4MCwgaGVpZ2h0OiA3MjAgfTtcblxuICB3b3JrZXI/OiBXb3JrZXI7XG4gIC8qKlxuICAgKiBJZiB0aGUgZ2l2ZW4gT2JzZXJ2YWJsZSBlbWl0cywgYW4gaW1hZ2Ugd2lsbCBiZSBjYXB0dXJlZCBhbmQgZW1pdHRlZCB0aHJvdWdoICdpbWFnZUNhcHR1cmUnIEV2ZW50RW1pdHRlclxuICAgKi9cbiAgQElucHV0KClcbiAgcHVibGljIHNldCB0cmlnZ2VyKHRyaWdnZXI6IE9ic2VydmFibGU8bnVtYmVyPikge1xuICAgIGlmICh0aGlzLnRyaWdnZXJTdWJzY3JpcHRpb24pIHtcbiAgICAgIHRoaXMudHJpZ2dlclN1YnNjcmlwdGlvbi51bnN1YnNjcmliZSgpO1xuICAgIH1cblxuICAgIC8vIFN1YnNjcmliZSB0byBldmVudHMgZnJvbSB0aGlzIE9ic2VydmFibGUgdG8gdGFrZSBzbmFwc2hvdHNcbiAgICB0aGlzLnRyaWdnZXJTdWJzY3JpcHRpb24gPSB0cmlnZ2VyLnN1YnNjcmliZSgodGltZSkgPT4ge1xuICAgICAgdGhpcy50YWtlU25hcHNob3QodGltZSk7XG4gICAgfSk7XG4gIH1cblxuICAvKipcbiAgICogSWYgdGhlIGdpdmVuIE9ic2VydmFibGUgZW1pdHMsIHRoZSBhY3RpdmUgd2ViY2FtIHdpbGwgYmUgc3dpdGNoZWQgdG8gdGhlIG9uZSBpbmRpY2F0ZWQgYnkgdGhlIGVtaXR0ZWQgdmFsdWUuXG4gICAqIEBwYXJhbSBzd2l0Y2hDYW1lcmEgSW5kaWNhdGVzIHdoaWNoIHdlYmNhbSB0byBzd2l0Y2ggdG9cbiAgICogICB0cnVlOiBjeWNsZSBmb3J3YXJkcyB0aHJvdWdoIGF2YWlsYWJsZSB3ZWJjYW1zXG4gICAqICAgZmFsc2U6IGN5Y2xlIGJhY2t3YXJkcyB0aHJvdWdoIGF2YWlsYWJsZSB3ZWJjYW1zXG4gICAqICAgc3RyaW5nOiBhY3RpdmF0ZSB0aGUgd2ViY2FtIHdpdGggdGhlIGdpdmVuIGlkXG4gICAqL1xuICBASW5wdXQoKSBzd2l0Y2hDYW1lcmE6IGFueTtcblxuICBASG9zdExpc3RlbmVyKCd3aW5kb3c6cmVzaXplJywgWyckZXZlbnQnXSlcbiAgb25SZXNpemUoKSB7XG4gICAvLyB0aGlzLnZpZGVvUmVhZHkubmV4dChmYWxzZSk7XG4gICAgdGhpcy5yZXNpemVTdGFnZSgpO1xuICB9XG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyBicmVha3BvaW50T2JzZXJ2ZXI6IEJyZWFrcG9pbnRPYnNlcnZlcixcbiAgICBwdWJsaWMgcGxhdGZvcm06IFBsYXRmb3JtLFxuICAgIHByaXZhdGUgY2Q6IENoYW5nZURldGVjdG9yUmVmLFxuICAgIHByaXZhdGUgem9uZTogTmdab25lLFxuXG4gICAgcHJpdmF0ZSBjYW1lcmFQcm92aWRlcjogTmd4U2NhbmRvY0NhbWVyYVByb3ZpZGVyXG4gICkge1xuICAgIHN1cGVyKCk7XG4gICAgLy8gaWYgKHR5cGVvZiBXb3JrZXIgIT09ICd1bmRlZmluZWQnKSB7XG4gICAgLy8gICAvLyBDcmVhdGUgYSBuZXdcbiAgICAvLyAgIHRoaXMud29ya2VyID0gbmV3IFdvcmtlcihcbiAgICAvLyAgICAgbmV3IFVSTCgnLi8uLi8uLi8uLi8uLi9hcHAvYXBwLndvcmtlcicsIGltcG9ydC5tZXRhLnVybClcbiAgICAvLyAgICk7XG4gICAgLy8gICB0aGlzLndvcmtlci5vbm1lc3NhZ2UgPSAoeyBkYXRhIH0pID0+IHtcbiAgICAvLyAgICAgY29uc29sZS5sb2coYHBhZ2UgZ290IG1lc3NhZ2U6ICR7ZGF0YX1gKTtcbiAgICAvLyAgIH07XG4gICAgLy8gICBjb25zb2xlLmxvZygnUE9TVCcpO1xuICAgIC8vICAgdGhpcy53b3JrZXIucG9zdE1lc3NhZ2UoJ2hlbGxvJyk7XG4gICAgLy8gfSBlbHNlIHtcbiAgICAvLyAgIC8vIFdlYiB3b3JrZXJzIGFyZSBub3Qgc3VwcG9ydGVkIGluIHRoaXMgZW52aXJvbm1lbnQuXG4gICAgLy8gICAvLyBZb3Ugc2hvdWxkIGFkZCBhIGZhbGxiYWNrIHNvIHRoYXQgeW91ciBwcm9ncmFtIHN0aWxsIGV4ZWN1dGVzIGNvcnJlY3RseS5cbiAgICAvLyB9XG4gIH1cblxuICAvKipcbiAgICogR2V0IE1lZGlhVHJhY2tDb25zdHJhaW50cyB0byByZXF1ZXN0IHN0cmVhbWluZyB0aGUgZ2l2ZW4gZGV2aWNlXG4gICAqIEBwYXJhbSBkZXZpY2VJZFxuICAgKiBAcGFyYW0gYmFzZU1lZGlhVHJhY2tDb25zdHJhaW50cyBiYXNlIGNvbnN0cmFpbnRzIHRvIG1lcmdlIGRldmljZUlkLWNvbnN0cmFpbnQgaW50b1xuICAgKiBAcmV0dXJuc1xuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgZ2V0TWVkaWFDb25zdHJhaW50c0ZvckRldmljZShcbiAgICBkZXZpY2VJZDogc3RyaW5nLFxuICAgIGJhc2VNZWRpYVRyYWNrQ29uc3RyYWludHM6IE1lZGlhVHJhY2tDb25zdHJhaW50c1xuICApOiBNZWRpYVRyYWNrQ29uc3RyYWludHMge1xuICAgIGNvbnN0IHJlc3VsdDogTWVkaWFUcmFja0NvbnN0cmFpbnRzID0gYmFzZU1lZGlhVHJhY2tDb25zdHJhaW50c1xuICAgICAgPyBiYXNlTWVkaWFUcmFja0NvbnN0cmFpbnRzXG4gICAgICA6IHRoaXMuREVGQVVMVF9WSURFT19PUFRJT05TO1xuICAgIGlmIChkZXZpY2VJZCkge1xuICAgICAgcmVzdWx0LmRldmljZUlkID0geyBleGFjdDogZGV2aWNlSWQgfTtcbiAgICB9XG5cbiAgICByZXR1cm4gcmVzdWx0O1xuICB9XG5cbiAgLyoqXG4gICAqIFRyaWVzIHRvIGhhcnZlc3QgdGhlIGRldmljZUlkIGZyb20gdGhlIGdpdmVuIG1lZGlhU3RyZWFtVHJhY2sgb2JqZWN0LlxuICAgKiBCcm93c2VycyBwb3B1bGF0ZSB0aGlzIG9iamVjdCBkaWZmZXJlbnRseTsgdGhpcyBtZXRob2QgdHJpZXMgc29tZSBkaWZmZXJlbnQgYXBwcm9hY2hlc1xuICAgKiB0byByZWFkIHRoZSBpZC5cbiAgICogQHBhcmFtIG1lZGlhU3RyZWFtVHJhY2tcbiAgICogQHJldHVybnMgZGV2aWNlSWQgaWYgZm91bmQgaW4gdGhlIG1lZGlhU3RyZWFtVHJhY2tcbiAgICovXG4gIHByaXZhdGUgc3RhdGljIGdldERldmljZUlkRnJvbU1lZGlhU3RyZWFtVHJhY2soXG4gICAgbWVkaWFTdHJlYW1UcmFjazogTWVkaWFTdHJlYW1UcmFja1xuICApOiBhbnkge1xuICAgIGlmIChcbiAgICAgIG1lZGlhU3RyZWFtVHJhY2suZ2V0U2V0dGluZ3MgJiZcbiAgICAgIG1lZGlhU3RyZWFtVHJhY2suZ2V0U2V0dGluZ3MoKSAmJlxuICAgICAgbWVkaWFTdHJlYW1UcmFjay5nZXRTZXR0aW5ncygpLmRldmljZUlkXG4gICAgKSB7XG4gICAgICByZXR1cm4gbWVkaWFTdHJlYW1UcmFjay5nZXRTZXR0aW5ncygpLmRldmljZUlkO1xuICAgIH0gZWxzZSBpZiAoXG4gICAgICBtZWRpYVN0cmVhbVRyYWNrLmdldENvbnN0cmFpbnRzICYmXG4gICAgICBtZWRpYVN0cmVhbVRyYWNrLmdldENvbnN0cmFpbnRzKCkgJiZcbiAgICAgIG1lZGlhU3RyZWFtVHJhY2suZ2V0Q29uc3RyYWludHMoKS5kZXZpY2VJZFxuICAgICkge1xuICAgICAgY29uc3QgZGV2aWNlSWRPYmo6IGFueSA9IG1lZGlhU3RyZWFtVHJhY2suZ2V0Q29uc3RyYWludHMoKS5kZXZpY2VJZDtcblxuICAgICAgcmV0dXJuIFdlYmNhbUNvbXBvbmVudC5nZXRWYWx1ZUZyb21Db25zdHJhaW5ET01TdHJpbmcoZGV2aWNlSWRPYmopO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBUcmllcyB0byBoYXJ2ZXN0IHRoZSBmYWNpbmdNb2RlIGZyb20gdGhlIGdpdmVuIG1lZGlhU3RyZWFtVHJhY2sgb2JqZWN0LlxuICAgKiBCcm93c2VycyBwb3B1bGF0ZSB0aGlzIG9iamVjdCBkaWZmZXJlbnRseTsgdGhpcyBtZXRob2QgdHJpZXMgc29tZSBkaWZmZXJlbnQgYXBwcm9hY2hlc1xuICAgKiB0byByZWFkIHRoZSB2YWx1ZS5cbiAgICogQHBhcmFtIG1lZGlhU3RyZWFtVHJhY2tcbiAgICogQHJldHVybnMgZmFjaW5nTW9kZSBpZiBmb3VuZCBpbiB0aGUgbWVkaWFTdHJlYW1UcmFja1xuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgZ2V0RmFjaW5nTW9kZUZyb21NZWRpYVN0cmVhbVRyYWNrKFxuICAgIG1lZGlhU3RyZWFtVHJhY2s6IE1lZGlhU3RyZWFtVHJhY2tcbiAgKTogYW55IHtcbiAgICBpZiAobWVkaWFTdHJlYW1UcmFjaykge1xuICAgICAgaWYgKFxuICAgICAgICBtZWRpYVN0cmVhbVRyYWNrLmdldFNldHRpbmdzICYmXG4gICAgICAgIG1lZGlhU3RyZWFtVHJhY2suZ2V0U2V0dGluZ3MoKSAmJlxuICAgICAgICBtZWRpYVN0cmVhbVRyYWNrLmdldFNldHRpbmdzKCkuZmFjaW5nTW9kZVxuICAgICAgKSB7XG4gICAgICAgIHJldHVybiBtZWRpYVN0cmVhbVRyYWNrLmdldFNldHRpbmdzKCkuZmFjaW5nTW9kZTtcbiAgICAgIH0gZWxzZSBpZiAoXG4gICAgICAgIG1lZGlhU3RyZWFtVHJhY2suZ2V0Q29uc3RyYWludHMgJiZcbiAgICAgICAgbWVkaWFTdHJlYW1UcmFjay5nZXRDb25zdHJhaW50cygpICYmXG4gICAgICAgIG1lZGlhU3RyZWFtVHJhY2suZ2V0Q29uc3RyYWludHMoKS5mYWNpbmdNb2RlXG4gICAgICApIHtcbiAgICAgICAgY29uc3QgZmFjaW5nTW9kZUNvbnN0cmFpbnQ6IGFueSA9XG4gICAgICAgICAgbWVkaWFTdHJlYW1UcmFjay5nZXRDb25zdHJhaW50cygpLmZhY2luZ01vZGU7XG4gICAgICAgIHJldHVybiBXZWJjYW1Db21wb25lbnQuZ2V0VmFsdWVGcm9tQ29uc3RyYWluRE9NU3RyaW5nKFxuICAgICAgICAgIGZhY2luZ01vZGVDb25zdHJhaW50XG4gICAgICAgICk7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIERldGVybWluZXMgd2hldGhlciB0aGUgZ2l2ZW4gbWVkaWFTdHJlYW1UcmFjayBjbGFpbXMgaXRzZWxmIGFzIHVzZXIgZmFjaW5nXG4gICAqIEBwYXJhbSBtZWRpYVN0cmVhbVRyYWNrXG4gICAqL1xuICBwcml2YXRlIHN0YXRpYyBpc1VzZXJGYWNpbmcobWVkaWFTdHJlYW1UcmFjazogTWVkaWFTdHJlYW1UcmFjayk6IGJvb2xlYW4ge1xuICAgIGNvbnN0IGZhY2luZ01vZGU6IHN0cmluZyA9XG4gICAgICBXZWJjYW1Db21wb25lbnQuZ2V0RmFjaW5nTW9kZUZyb21NZWRpYVN0cmVhbVRyYWNrKG1lZGlhU3RyZWFtVHJhY2spO1xuICAgIHJldHVybiBmYWNpbmdNb2RlID8gJ3VzZXInID09PSBmYWNpbmdNb2RlLnRvTG93ZXJDYXNlKCkgOiBmYWxzZTtcbiAgfVxuICBsYW5kc2NhcGUgPSB0cnVlO1xuICAvKipcbiAgICogRXh0cmFjdHMgdGhlIHZhbHVlIGZyb20gdGhlIGdpdmVuIENvbnN0cmFpbkRPTVN0cmluZ1xuICAgKiBAcGFyYW0gY29uc3RyYWluRE9NU3RyaW5nXG4gICAqL1xuICBwcml2YXRlIHN0YXRpYyBnZXRWYWx1ZUZyb21Db25zdHJhaW5ET01TdHJpbmcoY29uc3RyYWluRE9NU3RyaW5nOiBhbnkpOiBhbnkge1xuICAgIGlmIChjb25zdHJhaW5ET01TdHJpbmcpIHtcbiAgICAgIGlmIChjb25zdHJhaW5ET01TdHJpbmcgaW5zdGFuY2VvZiBTdHJpbmcpIHtcbiAgICAgICAgcmV0dXJuIFN0cmluZyhjb25zdHJhaW5ET01TdHJpbmcpO1xuICAgICAgfSBlbHNlIGlmIChcbiAgICAgICAgQXJyYXkuaXNBcnJheShjb25zdHJhaW5ET01TdHJpbmcpICYmXG4gICAgICAgIEFycmF5KGNvbnN0cmFpbkRPTVN0cmluZykubGVuZ3RoID4gMFxuICAgICAgKSB7XG4gICAgICAgIHJldHVybiBTdHJpbmcoY29uc3RyYWluRE9NU3RyaW5nWzBdKTtcbiAgICAgIH0gZWxzZSBpZiAodHlwZW9mIGNvbnN0cmFpbkRPTVN0cmluZyA9PT0gJ29iamVjdCcpIHtcbiAgICAgICAgaWYgKGNvbnN0cmFpbkRPTVN0cmluZ1snZXhhY3QnXSkge1xuICAgICAgICAgIHJldHVybiBTdHJpbmcoY29uc3RyYWluRE9NU3RyaW5nWydleGFjdCddKTtcbiAgICAgICAgfSBlbHNlIGlmIChjb25zdHJhaW5ET01TdHJpbmdbJ2lkZWFsJ10pIHtcbiAgICAgICAgICByZXR1cm4gU3RyaW5nKGNvbnN0cmFpbkRPTVN0cmluZ1snaWRlYWwnXSk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIHJlc2l6ZVN0YWdlKCkge1xuICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgdGhpcy51cGRhdGVTaXplKCk7XG4gICAgICB0aGlzLnVwZGF0ZWNhbnZhc1NpemUoKTtcbiAgICAgIHRoaXMuZHJhd1JlY3RhbmdsZSgpO1xuICAgICAgdGhpcy5jZC5kZXRlY3RDaGFuZ2VzKCk7XG4gICAgICB0aGlzLnZpZGVvUmVhZHkubmV4dCh0cnVlKTtcbiAgICB9LCAxMCk7XG4gIH1cblxuICBnZXQgY2FudmFzSGVpZ2h0KCkge1xuICAgIGNvbnN0IGxhbmRzY2FwZSA9IHRoaXMud2lkdGggPiB0aGlzLmhlaWdodDtcbiAgICBjb25zdCBhc3BlY3QgPSB0aGlzLnZpZGVvU2l6ZS53aWR0aCAvIHRoaXMudmlkZW9TaXplLmhlaWdodDtcbiAgICBpZiAoIWxhbmRzY2FwZSkge1xuICAgICAgcmV0dXJuIHRoaXMud2lkdGggKiBhc3BlY3Q7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLmhlaWdodDtcbiAgfVxuICB1cGRhdGVjYW52YXNTaXplKCkge1xuICAgLy8gY29uc29sZS5sb2codGhpcy53aWR0aCwgdGhpcy5oZWlnaHQpO1xuICAgIGNvbnN0IGxhbmRzY2FwZSA9IHRoaXMud2lkdGggPiB0aGlzLmhlaWdodDtcbiAgICBjb25zdCBhc3BlY3QgPSB0aGlzLnZpZGVvU2l6ZS53aWR0aCAvIHRoaXMudmlkZW9TaXplLmhlaWdodDtcblxuICAgIGxldCB3aWR0aCA9IHRoaXMud2lkdGg7XG4gICAgbGV0IGhlaWdodCA9IHRoaXMuaGVpZ2h0O1xuXG4gICAgaWYgKGxhbmRzY2FwZSkge1xuICAgICAgaGVpZ2h0ID0gd2lkdGggLyBhc3BlY3Q7XG4gICAgfSBlbHNlIHtcbiAgICAgIGhlaWdodCA9IHdpZHRoIC8gYXNwZWN0O1xuICAgIH1cblxuICAgIGlmIChoZWlnaHQgPiB0aGlzLmhlaWdodCkge1xuICAgICAgaGVpZ2h0ID0gdGhpcy5oZWlnaHQ7XG4gICAgICB3aWR0aCA9IHRoaXMuaGVpZ2h0ICogYXNwZWN0O1xuICAgIH1cblxuICAgIHRoaXMuY2FudmFzU2l6ZSA9IHsgd2lkdGg6IE1hdGgucm91bmQod2lkdGgpLCBoZWlnaHQ6IE1hdGgucm91bmQoaGVpZ2h0KSB9O1xuXG4gICAgY29uc29sZS5sb2codGhpcy5jYW52YXNTaXplLCB0aGlzLnZpZGVvU2l6ZSk7XG5cbiAgICBpZiAodGhpcy5pc01vYmlsZSkge1xuICAgICAgdGhpcy52aWRlby5uYXRpdmVFbGVtZW50LnNldEF0dHJpYnV0ZSgnaGVpZ2h0JywgdGhpcy5jYW52YXNTaXplLmhlaWdodCk7XG4gICAgICB0aGlzLnZpZGVvLm5hdGl2ZUVsZW1lbnQuc2V0QXR0cmlidXRlKCd3aWR0aCcsIHRoaXMuY2FudmFzU2l6ZS53aWR0aCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGlmICh0aGlzLnZpZGVvU2l6ZS53aWR0aCA+IHRoaXMudmlkZW9TaXplLmhpZ2h0KSB7XG4gICAgICAgIHRoaXMudmlkZW8ubmF0aXZlRWxlbWVudC5zZXRBdHRyaWJ1dGUoJ2hlaWdodCcsIHRoaXMuY2FudmFzU2l6ZS5oZWlnaHQpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy52aWRlby5uYXRpdmVFbGVtZW50LnNldEF0dHJpYnV0ZSgnd2lkdGgnLCB0aGlzLmNhbnZhc1NpemUud2lkdGgpO1xuICAgICAgfVxuICAgIH1cblxuICAgIHRoaXMuZHJhd1JlY3RhbmdsZSgpO1xuXG4gICAgdGhpcy5jZC5kZXRlY3RDaGFuZ2VzKCk7XG4gIH1cblxuICBwcml2YXRlIHNldHVwV29ya2VyKCkge1xuICAgIGlmICh0eXBlb2YgV29ya2VyICE9PSAndW5kZWZpbmVkJykge1xuICAgICAgLy8gQ3JlYXRlIGEgbmV3XG5cbiAgICAgIGlmICh0aGlzLnR5cGUgIT09ICdzZWxmaWUnKSB7XG4gICAgICAgIHRoaXMuem9uZS5ydW5PdXRzaWRlQW5ndWxhcigoKSA9PiB7XG4gICAgICAgICAgdmFyIGJsb2IgPSBuZXcgQmxvYihbd29ya2VydGV4dF0sIHsgdHlwZTogJ3RleHQvamF2YXNjcmlwdCcgfSk7XG5cbiAgICAgICAgICB2YXIgdXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChibG9iKTtcblxuICAgICAgICAgIHRoaXMud29ya2VyID0gbmV3IFdvcmtlcih1cmwpO1xuICAgICAgICAgIHRoaXMud29ya2VyLm9ubWVzc2FnZSA9ICh7IGRhdGEgfSkgPT4ge1xuICAgICAgICAgICAgaWYgKGRhdGEuYmFzZTY0KSB7XG5cbiAgICAgICAgICAgICAgZGF0YS50eXBlID09ICdkYXRhJztcbiAgICAgICAgICAgICAgdGhpcy5pbWFnZUhhbmRsZXIubmV4dChkYXRhKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9O1xuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICBwdWJsaWMgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIHRoaXMuc2V0dXBXb3JrZXIoKTtcblxuICAgIGlmICh0aGlzLmltYWdlSGFuZGxlcikge1xuICAgICAgdGhpcy5fX3N1YnModGhpcy5pbWFnZUhhbmRsZXIpLnN1YnNjcmliZSgocmVzcCkgPT4ge1xuICAgICAgICBpZiAocmVzcC50eXBlID09PSAnc3RvcCcpIHtcbiAgICAgICAgICB0aGlzLndvcmtlcj8ucG9zdE1lc3NhZ2UoeyB0eXBlOiAnc3RvcCcgfSk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH1cbiAgICB0aGlzLl9fc3Vicyh0aGlzLmJyZWFrcG9pbnRPYnNlcnZlclxuICAgICAgLm9ic2VydmUoW0JyZWFrcG9pbnRzLkhhbmRzZXRQb3J0cmFpdCwgQnJlYWtwb2ludHMuSGFuZHNldExhbmRzY2FwZV0pKVxuICAgICAgLnN1YnNjcmliZSgoc3RhdGU6IEJyZWFrcG9pbnRTdGF0ZSkgPT4ge1xuICAgICAgICBpZiAodGhpcy5wbGF0Zm9ybS5JT1MgfHwgdGhpcy5wbGF0Zm9ybS5BTkRST0lEKSB7XG4gICAgICAgICAgLy8gdGhpcy5sYW5kc2NhcGUgPSBzdGF0ZS5icmVha3BvaW50c1tCcmVha3BvaW50cy5IYW5kc2V0TGFuZHNjYXBlXTtcbiAgICAgICAgfVxuICAgICAgICBjb25zdCBtb2JpbGUgPSB0aGlzLnBsYXRmb3JtLklPUztcbiAgICAgICAgaWYgKHN0YXRlLm1hdGNoZXMgJiYgbW9iaWxlKSB7XG4gICAgICAgICAgdGhpcy51cGRhdGVTaXplKCk7XG4gICAgICAgICAgdGhpcy51cGRhdGVjYW52YXNTaXplKCk7XG4gICAgICAgICAgdGhpcy5kcmF3UmVjdGFuZ2xlKCk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIGlmICh0aGlzLnN3aXRjaENhbWVyYSkge1xuICAgICAgaWYgKHRoaXMuc3dpdGNoQ2FtZXJhU3Vic2NyaXB0aW9uKSB7XG4gICAgICAgIHRoaXMuc3dpdGNoQ2FtZXJhU3Vic2NyaXB0aW9uLnVuc3Vic2NyaWJlKCk7XG4gICAgICB9XG5cbiAgICAgIC8vIFN1YnNjcmliZSB0byBldmVudHMgZnJvbSB0aGlzIE9ic2VydmFibGUgdG8gc3dpdGNoIHZpZGVvIGRldmljZVxuICAgICAgdGhpcy5zd2l0Y2hDYW1lcmFTdWJzY3JpcHRpb24gPSB0aGlzLnN3aXRjaENhbWVyYS5zdWJzY3JpYmUoXG4gICAgICAgICh2YWx1ZTogc3RyaW5nKSA9PiB7XG4gICAgICAgICAgdGhpcy5zd2l0Y2hUb1ZpZGVvSW5wdXQodmFsdWUpO1xuICAgICAgICB9XG4gICAgICApO1xuICAgIH1cblxuICAgIHRoaXMuZGV0ZWN0QXZhaWxhYmxlRGV2aWNlcygpXG4gICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgIC8vIHN0YXJ0IHZpZGVvXG5cbiAgICAgICAgaWYgKHRoaXMuY2FuU3RhcnQpIHtcbiAgICAgICAgICB0aGlzLnN3aXRjaFRvVmlkZW9JbnB1dCgnJyk7XG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgICAuY2F0Y2goKGVycjogc3RyaW5nKSA9PiB7XG4gICAgICAgIHRoaXMuaW5pdEVycm9yLm5leHQoPFdlYmNhbUluaXRFcnJvcj57IG1lc3NhZ2U6IGVyciB9KTtcbiAgICAgICAgLy8gZmFsbGJhY2s6IHN0aWxsIHRyeSB0byBsb2FkIHdlYmNhbSwgZXZlbiBpZiBkZXZpY2UgZW51bWVyYXRpb24gZmFpbGVkXG4gICAgICAgIGlmICh0aGlzLmNhblN0YXJ0KSB7XG4gICAgICAgICAgdGhpcy5zd2l0Y2hUb1ZpZGVvSW5wdXQoJycpO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgfVxuXG4gIHB1YmxpYyBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICB0aGlzLmNhblN0YXJ0ID0gZmFsc2U7XG4gICAgdGhpcy5uYXRpdmVWaWRlb0VsZW1lbnQucGF1c2UoKTtcbiAgICB0aGlzLmRlc3Ryb3llZC5lbWl0KHRydWUpO1xuICAgIHRoaXMuc3RvcE1lZGlhVHJhY2tzKCk7XG4gICAgdGhpcy51bnN1YnNjcmliZUZyb21TdWJzY3JpcHRpb25zKCk7XG4gICAgdGhpcy5zaHV0ZG93biA9IHRydWU7XG5cbiAgICAvLyBzaHV0IGRvd24gd29ya2VyXG4gICAgaWYgKHRoaXMuaW1hZ2VIYW5kbGVyKSB7XG4gICAgICB0aGlzLmltYWdlSGFuZGxlci5uZXh0KHsgdHlwZTogJ3N0b3AnIH0pO1xuICAgIH1cblxuICAgIGlmICh0aGlzLm5hdGl2ZVZpZGVvRWxlbWVudC5zcmNPYmplY3QpIHtcbiAgICAgICgodGhpcy5uYXRpdmVWaWRlb0VsZW1lbnQgYXMgSFRNTFZpZGVvRWxlbWVudCkuc3JjT2JqZWN0IGFzIE1lZGlhU3RyZWFtKVxuICAgICAgICAuZ2V0VHJhY2tzKClcbiAgICAgICAgLmZvckVhY2goKHRyYWNrKSA9PiB7XG4gICAgICAgICAgdHJhY2suc3RvcCgpO1xuICAgICAgICB9KTtcbiAgICAgIHRoaXMubmF0aXZlVmlkZW9FbGVtZW50LnNyY09iamVjdCA9IG51bGw7XG4gICAgfVxuXG4gICAgdGhpcy5fX2Rlc3Ryb3koKTtcblxuICAgIC8vIHRoaXMubmF0aXZlVmlkZW9FbGVtZW50LnNyYyA9IG51bGw7XG4gICAgLy8gdGhpcy5uYXRpdmVWaWRlb0VsZW1lbnQuc3JjT2JqZWN0ID0gbnVsbDtcbiAgfVxuXG4gIHByaXZhdGUgdGFrZVNlbGZpZSgpIHtcbiAgICBjb25zdCBfdmlkZW8gPSB0aGlzLm5hdGl2ZVZpZGVvRWxlbWVudDtcblxuICAgIGNvbnN0IG1pbWVUeXBlOiBzdHJpbmcgPSB0aGlzLmltYWdlVHlwZVxuICAgICAgPyB0aGlzLmltYWdlVHlwZVxuICAgICAgOiBXZWJjYW1Db21wb25lbnQuREVGQVVMVF9JTUFHRV9UWVBFO1xuXG4gICAgbGV0IF9jYW52YXM6IEhUTUxDYW52YXNFbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnY2FudmFzJyk7IC8vdGhpcy5jYW52YXNTbmFwc2hvdC5uYXRpdmVFbGVtZW50O1xuXG4gICAgY29uc3QgeyB3aWR0aCwgaGVpZ2h0IH0gPSB0aGlzLnZpZGVvU2l6ZTtcbiAgIC8vIGNvbnNvbGUubG9nKHdpZHRoLCBoZWlnaHQpO1xuXG4gICAgY29uc3QgeyBwYWRkaW5nLCB0b3AsIHJXaWR0aCwgckhlaWdodCB9ID0gdGhpcy5zbmFwUmVjdGFuZ2xlO1xuICAgIF9jYW52YXMud2lkdGggPSByV2lkdGg7XG4gICAgX2NhbnZhcy5oZWlnaHQgPSBySGVpZ2h0O1xuXG4gICAgY29uc3QgY3R4OiBhbnkgPSBfY2FudmFzLmdldENvbnRleHQoJzJkJywge1xuICAgICAgYWxwaGE6IGZhbHNlLFxuICAgICAgcG93ZXJQcmVmZXJlbmNlOiAnaGlnaC1wZXJmb3JtYW5jZScsXG4gICAgICBhbnRpYWxpYXM6IGZhbHNlLFxuICAgICAgZGVwdGg6IGZhbHNlLFxuICAgICAgZGVzeW5jaHJvbml6ZWQ6IHRydWUsXG4gICAgfSk7XG5cbiAgICBpZiAoY3R4KSB7XG4gICAgICBjdHguZHJhd0ltYWdlKFxuICAgICAgICBfdmlkZW8sXG4gICAgICAgIHBhZGRpbmcsXG4gICAgICAgIHRvcCxcbiAgICAgICAgcldpZHRoLFxuICAgICAgICBySGVpZ2h0LFxuICAgICAgICAwLFxuICAgICAgICAwLFxuICAgICAgICByV2lkdGgsXG4gICAgICAgIHJIZWlnaHRcbiAgICAgICk7XG5cbiAgICAgIGNvbnN0IGltZ0FzVXJsOiBhbnkgPSBfY2FudmFzLnRvRGF0YVVSTChtaW1lVHlwZSwgMSk7XG5cbiAgICAgIHRoaXMuaW1hZ2VDYXB0dXJlLmVtaXQoXG4gICAgICAgIG5ldyBXZWJjYW1JbWFnZShpbWdBc1VybCwgbWltZVR5cGUsIG5ldyBJbWFnZURhdGEoMSwgMSksIG51bGwpXG4gICAgICApO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBUYWtlcyBhIHNuYXBzaG90IG9mIHRoZSBjdXJyZW50IHdlYmNhbSdzIHZpZXcgYW5kIGVtaXRzIHRoZSBpbWFnZSBhcyBhbiBldmVudFxuICAgKi9cblxuICBwdWJsaWMgdGFrZVNuYXBzaG90KHRpbWU6IG51bWJlcik6IHZvaWQge1xuICAgIGlmICh0aGlzLnR5cGUgPT09ICdzZWxmaWUnKSB7XG4gICAgICB0aGlzLnRha2VTZWxmaWUoKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpZiAodGhpcy50cmFja1Byb2Nlc3Nvcikge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aGlzLnpvbmUucnVuT3V0c2lkZUFuZ3VsYXIoKCkgPT4ge1xuICAgICAgY29uc3QgY2FudmFzU21hbFNpemUgPSAzODQ7XG5cbiAgICAgIGNvbnN0IF92aWRlbyA9IHRoaXMubmF0aXZlVmlkZW9FbGVtZW50O1xuXG4gICAgICBjb25zdCBtaW1lVHlwZTogc3RyaW5nID0gdGhpcy5pbWFnZVR5cGVcbiAgICAgICAgPyB0aGlzLmltYWdlVHlwZVxuICAgICAgICA6IFdlYmNhbUNvbXBvbmVudC5ERUZBVUxUX0lNQUdFX1RZUEU7XG5cbiAgICAgIGxldCBfY2FudmFzOiBIVE1MQ2FudmFzRWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ2NhbnZhcycpOyAvL3RoaXMuY2FudmFzU25hcHNob3QubmF0aXZlRWxlbWVudDtcblxuICAgICAgY29uc3QgeyB3aWR0aCwgaGVpZ2h0IH0gPSB0aGlzLnZpZGVvU2l6ZTtcblxuICAgICAgX2NhbnZhcy53aWR0aCA9IHdpZHRoO1xuICAgICAgX2NhbnZhcy5oZWlnaHQgPSBoZWlnaHQ7XG5cbiAgICAgIGNvbnN0IGN0eDogYW55ID0gX2NhbnZhcy5nZXRDb250ZXh0KCcyZCcsIHtcbiAgICAgICAgYWxwaGE6IGZhbHNlLFxuICAgICAgICBwb3dlclByZWZlcmVuY2U6ICdoaWdoLXBlcmZvcm1hbmNlJyxcbiAgICAgICAgYW50aWFsaWFzOiBmYWxzZSxcbiAgICAgICAgZGVwdGg6IGZhbHNlLFxuICAgICAgICBkZXN5bmNocm9uaXplZDogdHJ1ZSxcbiAgICAgIH0pO1xuXG4gICAgICBjb25zdCBjYW52YXMgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdjYW52YXMnKTsgLy8gbmVlZHMgYW4gaW5pdGlhbCBzaXplXG4gICAgICBjYW52YXMuaGVpZ2h0ID0gY2FudmFzU21hbFNpemU7XG4gICAgICBjYW52YXMud2lkdGggPSBjYW52YXNTbWFsU2l6ZTtcblxuICAgICAgY29uc3QgY3R4TzogYW55ID0gY2FudmFzLmdldENvbnRleHQoJzJkJywge1xuICAgICAgICBhbHBoYTogZmFsc2UsXG4gICAgICAgIHBvd2VyUHJlZmVyZW5jZTogJ2hpZ2gtcGVyZm9ybWFuY2UnLFxuICAgICAgICBhbnRpYWxpYXM6IGZhbHNlLFxuICAgICAgICBkZXB0aDogZmFsc2UsXG4gICAgICAgIGRlc3luY2hyb25pemVkOiB0cnVlLFxuICAgICAgfSk7XG5cbiAgICAgIGlmIChjdHggJiYgY3R4Tykge1xuICAgICAgICBjdHguaW1hZ2VTbW9vdGhpbmdFbmFibGVkID0gZmFsc2U7XG5cbiAgICAgICAgY3R4LmRyYXdJbWFnZShfdmlkZW8sIDAsIDApO1xuICAgICAgICBjdHhPLmRyYXdJbWFnZShfdmlkZW8sIDAsIDAsIGNhbnZhc1NtYWxTaXplLCBjYW52YXNTbWFsU2l6ZSk7XG5cbiAgICAgICAgY29uc3QgaW1EYXRhOiBhbnkgPSBjdHguZ2V0SW1hZ2VEYXRhKFxuICAgICAgICAgIDAsXG4gICAgICAgICAgMCxcbiAgICAgICAgICB0aGlzLnZpZGVvU2l6ZS53aWR0aCxcbiAgICAgICAgICB0aGlzLnZpZGVvU2l6ZS5oZWlnaHRcbiAgICAgICAgKTtcblxuICAgICAgICBjb25zdCByZXNpemUgPSBjYW52YXMudG9EYXRhVVJMKG1pbWVUeXBlLCAxKTtcblxuICAgICAgICBjb25zdCBjdXJyZW50VGltZSA9IG5ldyBEYXRlKCkuZ2V0VGltZSgpO1xuICAgICAgICBjb25zdCBkaWZmID0gY3VycmVudFRpbWUgLSB0aW1lO1xuXG4gICAgICAgIGNvbnN0IGRlbGF5ID0gZGlmZiA+IDEwMCA/IDAgOiAxMDAgLSBkaWZmO1xuXG4gICAgICAgIGNvbnN0IHRpbWVvdXQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICB0aGlzLmltYWdlQ2FwdHVyZS5lbWl0KG5ldyBXZWJjYW1JbWFnZSgnJywgbWltZVR5cGUsIGltRGF0YSwgcmVzaXplKSk7XG4gICAgICAgICAgY2xlYXJUaW1lb3V0KHRpbWVvdXQpO1xuICAgICAgICB9LCBkZWxheSk7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIHVwZGF0ZSgpIHt9XG5cbiAgcmVzaXplSW1hZ2UoYmFzZTY0ZGF0YTogYW55KTogT2JzZXJ2YWJsZTxhbnk+IHtcbiAgICByZXR1cm4gbmV3IE9ic2VydmFibGUoKG9ic2VydmVyKSA9PiB7XG4gICAgICBsZXQgY2FudmFzOiBhbnkgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdjYW52YXMnKTtcbiAgICAgIGNvbnN0IGN0eCA9IGNhbnZhcy5nZXRDb250ZXh0KCcyZCcpO1xuICAgICAgY29uc3QgbWF4X3NpemUgPSAzODQ7XG4gICAgICBsZXQgaW1hZ2U6IGFueSA9IG5ldyBJbWFnZSgpO1xuXG4gICAgICBpbWFnZS5zcmMgPSBiYXNlNjRkYXRhO1xuICAgICAgaW1hZ2Uub25sb2FkID0gKCkgPT4ge1xuICAgICAgICBsZXQgd2lkdGggPSBpbWFnZS53aWR0aDtcbiAgICAgICAgbGV0IGhlaWdodCA9IGltYWdlLmhlaWdodDtcblxuICAgICAgICBpZiAod2lkdGggPCBoZWlnaHQpIHtcbiAgICAgICAgICBoZWlnaHQgKj0gbWF4X3NpemUgLyB3aWR0aDtcbiAgICAgICAgICB3aWR0aCA9IG1heF9zaXplO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHdpZHRoICo9IG1heF9zaXplIC8gaGVpZ2h0O1xuICAgICAgICAgIGhlaWdodCA9IG1heF9zaXplO1xuICAgICAgICB9XG4gICAgICAgIGNhbnZhcy53aWR0aCA9IHdpZHRoO1xuICAgICAgICBjYW52YXMuaGVpZ2h0ID0gaGVpZ2h0O1xuXG4gICAgICAgIC8vIGNvbnNvbGUubG9nKHdpZHRoLCBoZWlnaHQpO1xuXG4gICAgICAgIGlmIChjdHgpIHtcbiAgICAgICAgICAvLyBjdHgucm90YXRlKCg5MCAqIE1hdGguUEkpIC8gMTgwKTtcbiAgICAgICAgICAvLyBjdHgudHJhbnNsYXRlKDAsIC1jYW52YXMud2lkdGgpO1xuICAgICAgICAgIGN0eC5kcmF3SW1hZ2UoaW1hZ2UsIDAsIDAsIDM4NCwgMzg0KTtcblxuICAgICAgICAgIG9ic2VydmVyLm5leHQoY2FudmFzLnRvRGF0YVVSTCgpKTtcbiAgICAgICAgICBjdHguY2xlYXJSZWN0KDAsIDAsIDEsIDEpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIG9ic2VydmVyLmVycm9yKHsgdHlwZTogJ2dlbmVyaWMgZXJyb3InIH0pO1xuICAgICAgICB9XG4gICAgICAgIGltYWdlID0gbnVsbDtcbiAgICAgICAgY2FudmFzID0gbnVsbDtcbiAgICAgIH07XG4gICAgICBpbWFnZS5vbmVycm9yID0gKGU6IGFueSkgPT4ge1xuICAgICAgICBvYnNlcnZlci5lcnJvcihlKTtcbiAgICAgIH07XG4gICAgfSk7XG4gIH1cblxuICAvKipcbiAgICogU3dpdGNoZXMgdG8gdGhlIG5leHQvcHJldmlvdXMgdmlkZW8gZGV2aWNlXG4gICAqIEBwYXJhbSBmb3J3YXJkXG4gICAqL1xuICBwdWJsaWMgcm90YXRlVmlkZW9JbnB1dChmb3J3YXJkOiBib29sZWFuKSB7XG4gICAgaWYgKHRoaXMuYXZhaWxhYmxlVmlkZW9JbnB1dHMgJiYgdGhpcy5hdmFpbGFibGVWaWRlb0lucHV0cy5sZW5ndGggPiAxKSB7XG4gICAgICBjb25zdCBpbmNyZW1lbnQ6IG51bWJlciA9IGZvcndhcmRcbiAgICAgICAgPyAxXG4gICAgICAgIDogdGhpcy5hdmFpbGFibGVWaWRlb0lucHV0cy5sZW5ndGggLSAxO1xuICAgICAgY29uc3QgbmV4dElucHV0SW5kZXggPVxuICAgICAgICAodGhpcy5hY3RpdmVWaWRlb0lucHV0SW5kZXggKyBpbmNyZW1lbnQpICVcbiAgICAgICAgdGhpcy5hdmFpbGFibGVWaWRlb0lucHV0cy5sZW5ndGg7XG4gICAgICB0aGlzLnN3aXRjaFRvVmlkZW9JbnB1dChcbiAgICAgICAgdGhpcy5hdmFpbGFibGVWaWRlb0lucHV0c1tuZXh0SW5wdXRJbmRleF0uZGV2aWNlSWRcbiAgICAgICk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFN3aXRjaGVzIHRoZSBjYW1lcmEtdmlldyB0byB0aGUgc3BlY2lmaWVkIHZpZGVvIGRldmljZVxuICAgKi9cbiAgcHVibGljIHN3aXRjaFRvVmlkZW9JbnB1dChkZXZpY2VJZDogc3RyaW5nKTogdm9pZCB7XG4gICAgLy8gaWYoZGV2aWNlSWQpe1xuICAgIHRoaXMudmlkZW9Jbml0aWFsaXplZCA9IGZhbHNlO1xuICAgIHRoaXMuc3RvcE1lZGlhVHJhY2tzKCk7XG4gICAgdGhpcy5pbml0V2ViY2FtKGRldmljZUlkLCB0aGlzLnZpZGVvT3B0aW9ucyk7XG4gICAgLy8gfVxuICB9XG5cbiAgLyoqXG4gICAqIEV2ZW50LWhhbmRsZXIgZm9yIHZpZGVvIHJlc2l6ZSBldmVudC5cbiAgICogVHJpZ2dlcnMgQW5ndWxhciBjaGFuZ2UgZGV0ZWN0aW9uIHNvIHRoYXQgbmV3IHZpZGVvIGRpbWVuc2lvbnMgZ2V0IGFwcGxpZWRcbiAgICovXG4gIHB1YmxpYyB2aWRlb1Jlc2l6ZSgpOiB2b2lkIHtcbiAgICAvLyBoZXJlIHRvIHRyaWdnZXIgQW5ndWxhciBjaGFuZ2UgZGV0ZWN0aW9uXG4gIH1cblxuICBwdWJsaWMgZ2V0IHZpZGVvV2lkdGgoKSB7XG4gICAgY29uc3QgdmlkZW9SYXRpbyA9IHRoaXMuZ2V0VmlkZW9Bc3BlY3RSYXRpbygpO1xuICAgIHJldHVybiBNYXRoLm1pbih0aGlzLndpZHRoLCB0aGlzLmhlaWdodCAqIHZpZGVvUmF0aW8pO1xuICB9XG5cbiAgcHVibGljIGdldCB2aWRlb0hlaWdodCgpIHtcbiAgICBjb25zdCB2aWRlb1JhdGlvID0gdGhpcy5nZXRWaWRlb0FzcGVjdFJhdGlvKCk7XG4gICAgcmV0dXJuIE1hdGgubWluKHRoaXMuaGVpZ2h0LCB0aGlzLndpZHRoIC8gdmlkZW9SYXRpbyk7XG4gIH1cblxuICBwdWJsaWMgZ2V0IHZpZGVvU3R5bGVDbGFzc2VzKCkge1xuICAgIGxldCBjbGFzc2VzOiBzdHJpbmcgPSAnJztcblxuICAgIGlmICh0aGlzLmlzTWlycm9ySW1hZ2UoKSkge1xuICAgICAgY2xhc3NlcyArPSAnbWlycm9yZWQgJztcbiAgICB9XG5cbiAgICByZXR1cm4gY2xhc3Nlcy50cmltKCk7XG4gIH1cblxuICBwdWJsaWMgZ2V0IG5hdGl2ZVZpZGVvRWxlbWVudCgpIHtcbiAgICByZXR1cm4gdGhpcy52aWRlby5uYXRpdmVFbGVtZW50O1xuICB9XG4gIC8vIHB1YmxpYyBnZXQgc21hbGxWaWRlb0VsZW1lbnQoKSB7XG4gIC8vICAgcmV0dXJuIHRoaXMudmlkZW9TbWFsbDtcbiAgLy8gfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIHRoZSB2aWRlbyBhc3BlY3QgcmF0aW8gb2YgdGhlIGFjdGl2ZSB2aWRlbyBzdHJlYW1cbiAgICovXG4gIHByaXZhdGUgZ2V0VmlkZW9Bc3BlY3RSYXRpbygpOiBudW1iZXIge1xuICAgIC8vIGNhbGN1bGF0ZSByYXRpbyBmcm9tIHZpZGVvIGVsZW1lbnQgZGltZW5zaW9ucyBpZiBwcmVzZW50XG4gICAgY29uc3QgdmlkZW9FbGVtZW50ID0gdGhpcy5uYXRpdmVWaWRlb0VsZW1lbnQ7XG4gICAgaWYgKFxuICAgICAgdmlkZW9FbGVtZW50LnZpZGVvV2lkdGggJiZcbiAgICAgIHZpZGVvRWxlbWVudC52aWRlb1dpZHRoID4gMCAmJlxuICAgICAgdmlkZW9FbGVtZW50LnZpZGVvSGVpZ2h0ICYmXG4gICAgICB2aWRlb0VsZW1lbnQudmlkZW9IZWlnaHQgPiAwXG4gICAgKSB7XG4gICAgICByZXR1cm4gdmlkZW9FbGVtZW50LnZpZGVvV2lkdGggLyB2aWRlb0VsZW1lbnQudmlkZW9IZWlnaHQ7XG4gICAgfVxuXG4gICAgLy8gbm90aGluZyBwcmVzZW50IC0gY2FsY3VsYXRlIHJhdGlvIGJhc2VkIG9uIHdpZHRoL2hlaWdodCBwYXJhbXNcbiAgICByZXR1cm4gdGhpcy53aWR0aCAvIHRoaXMuaGVpZ2h0O1xuICB9XG5cbiAgcHJpdmF0ZSB1cGRhdGVTaXplKCkge1xuICAgIGNvbnN0IHRyYWNrOiBNZWRpYVN0cmVhbVRyYWNrIHwgdW5kZWZpbmVkID1cbiAgICAgIHRoaXMubWVkaWFTdHJlYW0/LmdldFRyYWNrcygpWzBdO1xuXG4gICAgaWYgKHRyYWNrKSB7XG4gICAgICBsZXQgZGVzaXJlZDogYW55ID0ge1xuICAgICAgICB3aWR0aDoge1xuICAgICAgICAgIGlkZWFsOiAwLFxuICAgICAgICB9LFxuICAgICAgICBoZWlnaHQ6IHtcbiAgICAgICAgICBpZGVhbDogMCxcbiAgICAgICAgfSxcblxuICAgICAgICAvLyAgZmFjaW5nTW9kZTondXNlcicsXG4gICAgICAgIGZyYW1lUmF0ZTogeyBtaW46IDIwLCBpZGVhbDogMjQsIG1heDogMjQgfSxcbiAgICAgIH07XG4gICAgICBpZiAodHlwZW9mIHRyYWNrLmdldENhcGFiaWxpdGllcyA9PT0gJ2Z1bmN0aW9uJykge1xuICAgICAgICBjb25zdCB7IHdpZHRoLCBoZWlnaHQgfSA9IHRyYWNrLmdldENhcGFiaWxpdGllcygpO1xuICAgICAgICBkZXNpcmVkID0ge1xuICAgICAgICAgIHdpZHRoOiB7XG4gICAgICAgICAgICBpZGVhbDogdGhpcy5sYW5kc2NhcGUgJiYgdGhpcy5pc01vYmlsZSA/IGhlaWdodD8ubWF4IDogd2lkdGg/Lm1heCxcbiAgICAgICAgICB9LFxuICAgICAgICAgIGhlaWdodDoge1xuICAgICAgICAgICAgaWRlYWw6IHRoaXMubGFuZHNjYXBlICYmIHRoaXMuaXNNb2JpbGUgPyB3aWR0aD8ubWF4IDogaGVpZ2h0Py5tYXgsXG4gICAgICAgICAgfSxcblxuICAgICAgICAgIC8vICBmYWNpbmdNb2RlOid1c2VyJyxcbiAgICAgICAgICBmcmFtZVJhdGU6IHsgbWluOiAyMCwgaWRlYWw6IDI0LCBtYXg6IDI0IH0sXG4gICAgICAgIH07XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBkZXNpcmVkID0ge1xuICAgICAgICAgIHdpZHRoOiB7XG4gICAgICAgICAgICBpZGVhbDogMTkyMCxcbiAgICAgICAgICB9LFxuICAgICAgICAgIGhlaWdodDoge1xuICAgICAgICAgICAgaWRlYWw6IDEwODAsXG4gICAgICAgICAgfSxcblxuICAgICAgICAgIC8vICBmYWNpbmdNb2RlOid1c2VyJyxcbiAgICAgICAgICBmcmFtZVJhdGU6IHsgbWluOiAyMCwgaWRlYWw6IDMwLCBtYXg6IDMxIH0sXG4gICAgICAgIH07XG4gICAgICB9XG5cbi8vICBkZXNpcmVkLmhlaWdodC5pZGVhbCA9IDEwMDA7XG4vLyAgZGVzaXJlZC53aWR0aC5pZGVhbCA9IDEwMDA7XG5cbiAgICAgIGlmIChkZXNpcmVkLmhlaWdodC5pZGVhbCAmJiBkZXNpcmVkLmhlaWdodC5pZGVhbCA+PSAxNjAwKSB7XG4gICAgICAgIGRlc2lyZWQuaGVpZ2h0LmlkZWFsID0gMTYwMDtcbiAgICAgICAgZGVzaXJlZC53aWR0aC5pZGVhbCA9IDE2MDA7XG4gICAgICB9XG5cbiAgICAgIGlmICh0aGlzLnBsYXRmb3JtLkFORFJPSUQpIHtcbiAgICAgICAgLy9kZXNpcmVkLndpZHRoLmlkZWFsID0gMjE2MDtcbiAgICAgICAgLy9kZXNpcmVkLmhlaWdodC5pZGVhbCA9IDIxNjA7XG4gICAgICAgIC8vIGlmICghdGhpcy5sYW5kc2NhcGUpIHtcbiAgICAgICAgLy8gfSBlbHNlIHtcbiAgICAgICAgLy8gICBkZXNpcmVkLndpZHRoLmlkZWFsID0gd2lkdGg/Lm1heDtcbiAgICAgICAgLy8gICBkZXNpcmVkLmhlaWdodC5pZGVhbCA9IGhlaWdodD8ubWF4O1xuICAgICAgICAvLyB9XG4gICAgICB9XG5cbiAgICAgIHRyYWNrLmFwcGx5Q29uc3RyYWludHMoZGVzaXJlZCk7XG5cbiAgICAgIC8vdHJhY2suZ2V0U2V0dGluZ3MoKS5mYWNpbmdNb2RlIT09J3VzZXInO1xuXG4gICAgICB0aGlzLnZpZGVvU2l6ZSA9IHtcbiAgICAgICAgaGVpZ2h0OiB0aGlzLmxhbmRzY2FwZVxuICAgICAgICAgID8gdHJhY2suZ2V0U2V0dGluZ3MoKS5oZWlnaHRcbiAgICAgICAgICA6IHRyYWNrLmdldFNldHRpbmdzKCkud2lkdGgsXG4gICAgICAgIHdpZHRoOiB0aGlzLmxhbmRzY2FwZVxuICAgICAgICAgID8gdHJhY2suZ2V0U2V0dGluZ3MoKS53aWR0aFxuICAgICAgICAgIDogdHJhY2suZ2V0U2V0dGluZ3MoKS5oZWlnaHQsXG4gICAgICB9O1xuXG4gICAgICB0aGlzLmNkLmRldGVjdENoYW5nZXMoKTtcbiAgICB9XG4gIH1cblxuICBnZXRTdHJlYW1UcmFjayhzdHJlYW06IE1lZGlhU3RyZWFtKSB7XG4gICAgcmV0dXJuIHN0cmVhbS5nZXRWaWRlb1RyYWNrcygpWzBdO1xuICB9XG5cbiAgZ2V0VHJhY2tTZXR0aW5ncygpIHt9XG5cbiAgYWNjZXNWaWRlb1RyYWNrKHZpZGVvVHJhY2tDb25zdHJhaW50czogTWVkaWFTdHJlYW1Db25zdHJhaW50cyk6IGFueSB7XG4gICAgcmV0dXJuIGZyb20oXG4gICAgICBuYXZpZ2F0b3IubWVkaWFEZXZpY2VzLmdldFVzZXJNZWRpYSh2aWRlb1RyYWNrQ29uc3RyYWludHMpXG4gICAgKS5waXBlKFxuICAgICAgc3dpdGNoTWFwKChzdHJlYW06IE1lZGlhU3RyZWFtKSA9PiB7XG4gICAgICAgIC8vIGRlZmF1bHQgcmVzb2x1dGlvbiAxMjgweDcyMCwgY2hlY2sgbWF4IHJlc29sdXRpb25cblxuICAgICAgICBjb25zdCB0cmFjayA9IHRoaXMuZ2V0U3RyZWFtVHJhY2soc3RyZWFtKTtcblxuICAgICAgICBjb25zdCBjYXBhYmlsaXRpZXMgPSB0cmFjay5nZXRDYXBhYmlsaXRpZXMoKTtcblxuICAgICAgICBjb25zdCB7IGZhY2luZ01vZGUsIGhlaWdodCwgd2lkdGggfSA9IGNhcGFiaWxpdGllcztcblxuICAgICAgICBjb25zdCBkZXNpcmVkID0ge1xuICAgICAgICAgIHdpZHRoOiB7XG4gICAgICAgICAgICBtaW46IHRoaXMubGFuZHNjYXBlICYmIHRoaXMuaXNNb2JpbGUgPyBoZWlnaHQ/Lm1heCA6IHdpZHRoPy5tYXgsXG4gICAgICAgICAgfSxcbiAgICAgICAgICAvLyBoZWlnaHQ6IHtcbiAgICAgICAgICAvLyAgIGV4YWN0OiB0aGlzLmxhbmRzY2FwZSAmJiB0aGlzLmlzTW9iaWxlID8gd2lkdGg/Lm1heCA6IGhlaWdodD8ubWF4LFxuICAgICAgICAgIC8vIH0sXG4gICAgICAgICAgLy8gZmFjaW5nTW9kZTonZW4nLFxuICAgICAgICAgIC8vIGZyYW1lUmF0ZTogeyBtaW46IDI1LCBpZGVhbDogMzAsIG1heDogMzEgfSxcbiAgICAgICAgfTtcbiAgICAgICAgLy8gZ2V0IG1heCB3aWR0aFxuXG4gICAgICAgIGNvbnN0IHsgd2lkdGg6IHNXaWR0aCwgaGVpZ2h0OiBzSGVpZ2h0IH0gPSB0cmFjay5nZXRTZXR0aW5ncygpO1xuXG4gICAgICAgIGlmIChkZXNpcmVkLndpZHRoLm1pbiAhPSBzV2lkdGgpIHtcbiAgICAgICAgICByZXR1cm4gdGhpcy5hY2Nlc1ZpZGVvVHJhY2soeyB2aWRlbzogZGVzaXJlZCB9KTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vcmV0dXJuIHRoaXMuYWNjZXNWaWRlb1RyYWNrKHt2aWRlbzpkZXNpcmVkfSk7XG5cbiAgICAgICAgcmV0dXJuIG9mKHN0cmVhbSk7XG4gICAgICB9KVxuICAgICk7XG4gIH1cblxuICBwcml2YXRlIGRyYXdSZWN0YW5nbGUoKSB7XG4gICAgY29uc3QgX2NhbnZhcyA9IHRoaXMuY2FudmFzLm5hdGl2ZUVsZW1lbnQ7XG5cbiAgICBjb25zdCBjdHggPSBfY2FudmFzLmdldENvbnRleHQoJzJkJyk7XG4gICAgY3R4LmNsZWFyUmVjdCgwLCAwLCBfY2FudmFzLndpZHRoLCBfY2FudmFzLmhlaWdodCk7XG5cbiAgICAvLyBjdHguZHJhd0ltYWdlKF92aWRlbywgMCwgMCwgX2NhbnZhcy53aWR0aCwgX2NhbnZhcy5oZWlnaHQpO1xuXG4gICAgY29uc3QgeyBwYWRkaW5nLCB0b3AsIHJXaWR0aCwgckhlaWdodCB9ID0gdGhpcy5jYXJkUmVjdGFuZ2xlO1xuXG4gICAgY3R4LnN0cm9rZVN0eWxlID0gJ3JlZCc7XG5cbiAgICBjdHguc3Ryb2tlUmVjdChwYWRkaW5nLCB0b3AsIHJXaWR0aCwgckhlaWdodCk7XG4gIH1cblxuICBnZXRNYXhBdmFpbGFibGVSZXNvbHV0aW9uKCkge31cbiAgLyoqXG4gICAqIEluaXQgd2ViY2FtIGxpdmUgdmlld1xuICAgKi9cbiAgcHJpdmF0ZSBpbml0V2ViY2FtKFxuICAgIGRldmljZUlkOiBzdHJpbmcsXG4gICAgdXNlclZpZGVvVHJhY2tDb25zdHJhaW50czogTWVkaWFUcmFja0NvbnN0cmFpbnRzXG4gICkge1xuICAgIGNvbnNvbGUubG9nKCdJTklUIFdFQkNBTScpXG4gICAgY29uc3QgX3ZpZGVvID0gdGhpcy5uYXRpdmVWaWRlb0VsZW1lbnQ7XG4gICAgLy8gY29uc3QgdmlkZW9TdHJlYW1lciA9IHRoaXMudmlkZW9TdHJlYW1lci5uYXRpdmVFbGVtZW50O1xuICAgIC8vIGNvbnN0IHZpZGVvU21hbGwgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCd2aWRlbycpO1xuICAgIGlmIChuYXZpZ2F0b3IubWVkaWFEZXZpY2VzICYmIG5hdmlnYXRvci5tZWRpYURldmljZXMuZ2V0VXNlck1lZGlhKSB7XG4gICAgICAvLyBtZXJnZSBkZXZpY2VJZCAtPiB1c2VyVmlkZW9UcmFja0NvbnN0cmFpbnRzXG4gICAgICBjb25zdCB2aWRlb1RyYWNrQ29uc3RyYWludHMgPVxuICAgICAgICBXZWJjYW1Db21wb25lbnQuZ2V0TWVkaWFDb25zdHJhaW50c0ZvckRldmljZShcbiAgICAgICAgICBkZXZpY2VJZCxcbiAgICAgICAgICB1c2VyVmlkZW9UcmFja0NvbnN0cmFpbnRzXG4gICAgICAgICk7XG5cbiAgICAgIC8vIC8vIGlmIChkZXZpY2VJZCkge1xuICAgICAgLy8gdGhpcy5hY2Nlc1ZpZGVvVHJhY2soeyB2aWRlbzogdmlkZW9UcmFja0NvbnN0cmFpbnRzIH0pLnN1YnNjcmliZShcbiAgICAgIC8vICAgKHN0cmVhbTogTWVkaWFTdHJlYW0pID0+IHtcbiAgICAgIC8vICAgICBjb25zb2xlLmxvZyhcbiAgICAgIC8vICAgICAgICdWQUxJRCBTRVRUSU5HUycsXG4gICAgICAvLyAgICAgICBzdHJlYW0uZ2V0VmlkZW9UcmFja3MoKVswXS5nZXRTZXR0aW5ncygpXG4gICAgICAvLyAgICAgKTtcbiAgICAgIC8vICAgfVxuICAgICAgLy8gKTtcbiAgICAgIC8vIH1cbiAgICAgIGlmICh0aGlzLnBsYXRmb3JtLkFORFJPSUQpIHtcbiAgICAgICAgdmlkZW9UcmFja0NvbnN0cmFpbnRzLndpZHRoID0gMTYwMDtcbiAgICAgICAgdmlkZW9UcmFja0NvbnN0cmFpbnRzLmhlaWdodCA9IDE2MDA7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGNvbnN0cmFpbnRzID0ge1xuICAgICAgICBleGFjdDogZGV2aWNlSWQsXG4gICAgICAgIHdpZHRoOiB7IG1pbjogNDgwLCBpZGVhbDogMTI4MCB9LFxuICAgICAgICBoZWlnaHQ6IHsgbWluOiA0ODAsIGlkZWFsOiA3MjAgfSxcbiAgICAgICAgLy8gYXNwZWN0UmF0aW86IDMgLyAyLFxuICAgICAgICBmcmFtZVJhdGU6IHsgbWluOiAyMCB9LFxuICAgICAgfTtcblxuICAgICAgbmF2aWdhdG9yLm1lZGlhRGV2aWNlc1xuICAgICAgICAuZ2V0VXNlck1lZGlhKDxNZWRpYVN0cmVhbUNvbnN0cmFpbnRzPnsgdmlkZW86IHZpZGVvVHJhY2tDb25zdHJhaW50cyB9KVxuICAgICAgICAudGhlbigoc3RyZWFtOiBNZWRpYVN0cmVhbSkgPT4ge1xuICAgICAgICAgIHRoaXMubWVkaWFTdHJlYW0gPSBzdHJlYW07XG4gICAgICAgICAgX3ZpZGVvLnNyY09iamVjdCA9IHN0cmVhbTtcblxuICAgICAgICAgIHRoaXMudXBkYXRlU2l6ZSgpO1xuICAgICAgICAgIHRoaXMudXBkYXRlY2FudmFzU2l6ZSgpO1xuXG4gICAgICAgICAgX3ZpZGVvLnBsYXkoKTtcblxuICAgICAgICAgIF92aWRlby5vbnBsYXkgPSAoKSA9PiB7XG4gICAgICAgICAgICB0aGlzLnNldEFjdGl2ZUNhbWVyYShzdHJlYW0pO1xuICAgICAgICAgICAgdGhpcy5kcmF3UmVjdGFuZ2xlKCk7XG4gICAgICAgICAgfTtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnI6IGFueSkgPT4ge1xuICAgICAgICAgIGNvbnNvbGUubG9nKGVycik7XG4gICAgICAgICAgdGhpcy5pbml0RXJyb3IubmV4dCg8V2ViY2FtSW5pdEVycm9yPntcbiAgICAgICAgICAgIG1lc3NhZ2U6IGVyci5tZXNzYWdlLFxuICAgICAgICAgICAgbWVkaWFTdHJlYW1FcnJvcjogZXJyLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9KTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5pbml0RXJyb3IubmV4dCg8V2ViY2FtSW5pdEVycm9yPntcbiAgICAgICAgbWVzc2FnZTogJ0Nhbm5vdCByZWFkIFVzZXJNZWRpYSBmcm9tIE1lZGlhRGV2aWNlcy4nLFxuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgZ2V0IGlzTW9iaWxlKCkge1xuICAgIHJldHVybiB0aGlzLnBsYXRmb3JtLkFORFJPSUQgfHwgdGhpcy5wbGF0Zm9ybS5JT1M7XG4gIH1cblxuICBnZXQgY2FyZFJlY3RhbmdsZSgpIHtcbiAgICBjb25zdCBfY2FudmFzID0gdGhpcy5jYW52YXMubmF0aXZlRWxlbWVudDtcbiAgICBjb25zdCBkb2NTaXplID0gdGhpcy50eXBlID09PSAnc2VsZmllJyA/IDEgOiA4NiAvIDU1O1xuXG4gICAvLyBjb25zb2xlLmxvZyhfY2FudmFzLndpZHRoLCBfY2FudmFzLmhlaWdodCk7XG4gICAgbGV0IHBhZGRpbmcgPSAxMDtcblxuICAgIGxldCByV2lkdGggPSBfY2FudmFzLndpZHRoIC0gMiAqIHBhZGRpbmc7XG4gICAgbGV0IHJIZWlnaHQgPSByV2lkdGggLyBkb2NTaXplO1xuICAgIGxldCB0b3AgPSAoX2NhbnZhcy5oZWlnaHQgLSBySGVpZ2h0KSAvIDI7XG5cbiAgICBpZiAoKCF0aGlzLmlzTW9iaWxlICYmIF9jYW52YXMud2lkdGghPT0gX2NhbnZhcy5oZWlnaHQpKSB7XG4gICAgICBwYWRkaW5nID0gNDA7XG5cbiAgICAgIHJIZWlnaHQgPSBfY2FudmFzLmhlaWdodCAtIDIgKiBwYWRkaW5nO1xuICAgICAgcldpZHRoID0gckhlaWdodCAqIGRvY1NpemU7XG4gICAgICB0b3AgPSAoX2NhbnZhcy5oZWlnaHQgLSBySGVpZ2h0KSAvIDI7XG4gICAgICBwYWRkaW5nID0gKF9jYW52YXMud2lkdGggLSByV2lkdGgpIC8gMjtcbiAgICB9XG5cbiAgIC8vIGNvbnNvbGUubG9nKHBhZGRpbmcsIHRvcCwgcldpZHRoLCBySGVpZ2h0KTtcblxuICAgIHJldHVybiB7IHBhZGRpbmcsIHRvcCwgcldpZHRoLCBySGVpZ2h0IH07XG4gIH1cbiAgZ2V0IHNuYXBSZWN0YW5nbGUoKSB7XG4gICAgY29uc3QgX2NhbnZhcyA9IHRoaXMuY2FudmFzLm5hdGl2ZUVsZW1lbnQ7XG5cbiAgICBjb25zdCBhciA9IHRoaXMudmlkZW9TaXplLndpZHRoIC8gX2NhbnZhcy53aWR0aDtcblxuICAgIGxldCB7IHBhZGRpbmcsIHRvcCwgcldpZHRoLCBySGVpZ2h0IH0gPSB0aGlzLmNhcmRSZWN0YW5nbGU7XG5cbiAgICBwYWRkaW5nID0gcGFkZGluZyAqIGFyO1xuICAgIHRvcCA9IHRvcCAqIGFyO1xuICAgIHJXaWR0aCA9IHJXaWR0aCAqIGFyO1xuICAgIHJIZWlnaHQgPSBySGVpZ2h0ICogYXI7XG5cbiAgICByZXR1cm4geyBwYWRkaW5nLCB0b3AsIHJXaWR0aCwgckhlaWdodCB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzZXRBY3RpdmVDYW1lcmEoc3RyZWFtOiBhbnkpIHtcbiAgICB0aGlzLnZpZGVvUmVhZHkubmV4dChmYWxzZSk7XG4gICAgdGhpcy5zaG93VmlkZW8gPSBmYWxzZTtcbiAgICB0aGlzLmFjdGl2ZVZpZGVvU2V0dGluZ3MgPSBzdHJlYW0uZ2V0VmlkZW9UcmFja3MoKVswXS5nZXRTZXR0aW5ncygpO1xuICAgIGNvbnN0IGFjdGl2ZURldmljZUlkOiBzdHJpbmcgPVxuICAgICAgV2ViY2FtQ29tcG9uZW50LmdldERldmljZUlkRnJvbU1lZGlhU3RyZWFtVHJhY2soXG4gICAgICAgIHN0cmVhbS5nZXRWaWRlb1RyYWNrcygpWzBdXG4gICAgICApO1xuXG4gICAgY29uc3QgdmlkZW9UcmFjayA9IHN0cmVhbS5nZXRUcmFja3MoKVswXTtcbiAgICB0aGlzLnRyYWNrU2V0dGluZ3MgPSB2aWRlb1RyYWNrLmdldFNldHRpbmdzKCk7XG5cbiAgICB0aGlzLmNhbWVyYVByb3ZpZGVyLmNhbWVyYVdhc1N3aXRjaGVkKGFjdGl2ZURldmljZUlkKTtcblxuICAgIHRoaXMuY2FtZXJhU3dpdGNoZWQubmV4dChhY3RpdmVEZXZpY2VJZCk7XG5cbiAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIHRoaXMuem9uZS5ydW4oKCkgPT4ge1xuICAgICAgICB0aGlzLnJlc2l6ZVN0YWdlKCk7XG4gICAgICAgIHRoaXMuc2hvd1ZpZGVvID0gdHJ1ZTtcbiAgICAgICAgdGhpcy52aWRlb1JlYWR5Lm5leHQodHJ1ZSk7XG5cbiAgICAgICAgaWYgKE1lZGlhU3RyZWFtVHJhY2tQcm9jZXNzb3IpIHtcbiAgICAgICAgICB0aGlzLnRyYWNrUHJvY2Vzc29yID0gbmV3IE1lZGlhU3RyZWFtVHJhY2tQcm9jZXNzb3IodmlkZW9UcmFjayk7XG4gICAgICAgICAgbGV0IGZyYW1lU3RyZWFtID0gdGhpcy50cmFja1Byb2Nlc3Nvci5yZWFkYWJsZTtcbiAgICAgICAgICBpZiAoIXRoaXMuc2h1dGRvd24pIHtcbiAgICAgICAgICAgIHRoaXMud29ya2VyPy5wb3N0TWVzc2FnZShcbiAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgIHR5cGU6ICdzdGFydCcsXG5cbiAgICAgICAgICAgICAgICBmcmFtZVN0cmVhbTogZnJhbWVTdHJlYW0sXG4gICAgICAgICAgICAgICAgdHJhY2tTZXR0aW5nczogdGhpcy50cmFja1NldHRpbmdzLFxuICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICBbZnJhbWVTdHJlYW1dXG4gICAgICAgICAgICApO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgfSwgNTAwKTtcbiAgfVxuXG4gIHByaXZhdGUgZ2V0QWN0aXZlVmlkZW9UcmFjaygpOiBNZWRpYVN0cmVhbVRyYWNrIHwgYW55IHtcbiAgICByZXR1cm4gdGhpcy5tZWRpYVN0cmVhbSA/IHRoaXMubWVkaWFTdHJlYW0uZ2V0VmlkZW9UcmFja3MoKVswXSA6IG51bGw7XG4gIH1cblxuICBwcml2YXRlIGlzTWlycm9ySW1hZ2UoKTogYm9vbGVhbiB7XG4gICAgaWYgKCF0aGlzLmdldEFjdGl2ZVZpZGVvVHJhY2soKSkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cblxuICAgIC8vIGNoZWNrIGZvciBleHBsaWNpdCBtaXJyb3Igb3ZlcnJpZGUgcGFyYW1ldGVyXG4gICAge1xuICAgICAgbGV0IG1pcnJvcjogc3RyaW5nID0gJ2F1dG8nO1xuICAgICAgaWYgKHRoaXMubWlycm9ySW1hZ2UpIHtcbiAgICAgICAgaWYgKHR5cGVvZiB0aGlzLm1pcnJvckltYWdlID09PSAnc3RyaW5nJykge1xuICAgICAgICAgIG1pcnJvciA9IFN0cmluZyh0aGlzLm1pcnJvckltYWdlKS50b0xvd2VyQ2FzZSgpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIFdlYmNhbU1pcnJvclByb3BlcnRpZXNcbiAgICAgICAgICBpZiAodGhpcy5taXJyb3JJbWFnZS54KSB7XG4gICAgICAgICAgICBtaXJyb3IgPSB0aGlzLm1pcnJvckltYWdlLngudG9Mb3dlckNhc2UoKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgc3dpdGNoIChtaXJyb3IpIHtcbiAgICAgICAgY2FzZSAnYWx3YXlzJzpcbiAgICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgICAgY2FzZSAnbmV2ZXInOlxuICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBkZWZhdWx0OiBlbmFibGUgbWlycm9yaW5nIGlmIHdlYmNhbSBpcyB1c2VyIGZhY2luZ1xuICAgIHJldHVybiBXZWJjYW1Db21wb25lbnQuaXNVc2VyRmFjaW5nKHRoaXMuZ2V0QWN0aXZlVmlkZW9UcmFjaygpKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTdG9wcyBhbGwgYWN0aXZlIG1lZGlhIHRyYWNrcy5cbiAgICogVGhpcyBwcmV2ZW50cyB0aGUgd2ViY2FtIGZyb20gYmVpbmcgaW5kaWNhdGVkIGFzIGFjdGl2ZSxcbiAgICogZXZlbiBpZiBpdCBpcyBubyBsb25nZXIgdXNlZCBieSB0aGlzIGNvbXBvbmVudC5cbiAgICovXG4gIHByaXZhdGUgc3RvcE1lZGlhVHJhY2tzKCkge1xuICAgIC8vIHRoaXMudmlkZW8ubmF0aXZlRWxlbWVudC5wYXVzZSgpO1xuICAgIGlmICh0aGlzLnZpZGVvLm5hdGl2ZUVsZW1lbnQuc3JjT2JqZWN0KSB7XG4gICAgICAoKHRoaXMudmlkZW8ubmF0aXZlRWxlbWVudCBhcyBIVE1MVmlkZW9FbGVtZW50KS5zcmNPYmplY3QgYXMgTWVkaWFTdHJlYW0pXG4gICAgICAgIC5nZXRUcmFja3MoKVxuICAgICAgICAuZm9yRWFjaCgodHJhY2spID0+IHtcbiAgICAgICAgICB0cmFjay5zdG9wKCk7XG4gICAgICAgIH0pO1xuICAgICAgdGhpcy52aWRlby5uYXRpdmVFbGVtZW50LnNyY09iamVjdCA9IG51bGw7XG4gICAgICB0aGlzLnZpZGVvLm5hdGl2ZUVsZW1lbnQuc3JjID0gJyc7XG4gICAgfVxuXG4gICAgaWYgKHRoaXMubWVkaWFTdHJlYW0gJiYgdGhpcy5tZWRpYVN0cmVhbS5nZXRUcmFja3MpIHtcbiAgICAgIC8vIGdldFRyYWNrcygpIHJldHVybnMgYWxsIG1lZGlhIHRyYWNrcyAodmlkZW8rYXVkaW8pXG5cbiAgICAgIHRoaXMubWVkaWFTdHJlYW1cbiAgICAgICAgLmdldFRyYWNrcygpXG4gICAgICAgIC5mb3JFYWNoKCh0cmFjazogTWVkaWFTdHJlYW1UcmFjaykgPT4gdHJhY2suc3RvcCgpKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogVW5zdWJzY3JpYmUgZnJvbSBhbGwgb3BlbiBzdWJzY3JpcHRpb25zXG4gICAqL1xuICBwcml2YXRlIHVuc3Vic2NyaWJlRnJvbVN1YnNjcmlwdGlvbnMoKSB7XG4gICAgaWYgKHRoaXMudHJpZ2dlclN1YnNjcmlwdGlvbikge1xuICAgICAgdGhpcy50cmlnZ2VyU3Vic2NyaXB0aW9uLnVuc3Vic2NyaWJlKCk7XG4gICAgfVxuICAgIGlmICh0aGlzLnN3aXRjaENhbWVyYVN1YnNjcmlwdGlvbikge1xuICAgICAgdGhpcy5zd2l0Y2hDYW1lcmFTdWJzY3JpcHRpb24udW5zdWJzY3JpYmUoKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUmVhZHMgYXZhaWxhYmxlIGlucHV0IGRldmljZXNcbiAgICovXG4gIHByaXZhdGUgZGV0ZWN0QXZhaWxhYmxlRGV2aWNlcygpOiBQcm9taXNlPE1lZGlhRGV2aWNlSW5mb1tdPiB7XG4gICAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgIFdlYmNhbVV0aWwuZ2V0QXZhaWxhYmxlVmlkZW9JbnB1dHMoKS5zdWJzY3JpYmUoXG4gICAgICAgIChkZXZpY2VzOiBNZWRpYURldmljZUluZm9bXSkgPT4ge1xuICAgICAgICAgIHRoaXMuYXZhaWxhYmxlVmlkZW9JbnB1dHMgPSBkZXZpY2VzO1xuICAgICAgICAgIHJlc29sdmUoZGV2aWNlcyk7XG4gICAgICAgIH0sXG4gICAgICAgIChlcnIpID0+IHtcbiAgICAgICAgICB0aGlzLmF2YWlsYWJsZVZpZGVvSW5wdXRzID0gW107XG4gICAgICAgICAgcmVqZWN0KGVycik7XG4gICAgICAgIH1cbiAgICAgICk7XG4gICAgfSk7XG4gIH1cbn1cbiIsIjxkaXYgY2xhc3M9XCJ3ZWJjYW0td3JhcHBlclwiPlxuICA8ZGl2PlxuICAgIDx2aWRlb1xuICAgICAgI3ZpZGVvXG4gICAgICBhdXRvcGxheVxuICAgICAgbXV0ZWRcbiAgICAgIHN0eWxlPVwiZGlzcGxheTogYmxvY2tcIlxuICAgICAgcGxheXNpbmxpbmVcblxuICAgID48L3ZpZGVvPlxuXG4gICAgPGRpdiBjbGFzcz1cInJlY3RhbmdsZVwiPlxuXG4gICAgICA8Y2FudmFzXG4gICAgICAgICNjYW52YXNcbiAgICAgICAgW25nU3R5bGVdPVwieyB2aXNpYmlsaXR5OiBzaG93VmlkZW8gPyAndmlzaWJsZScgOiAnaGlkZGVuJyB9XCJcbiAgICAgICAgW3dpZHRoXT1cImNhbnZhc1NpemUud2lkdGhcIlxuICAgICAgICBbaGVpZ2h0XT1cImNhbnZhc1NpemUuaGVpZ2h0XCJcbiAgICAgID48L2NhbnZhcz5cbiAgICA8L2Rpdj5cbiAgPC9kaXY+XG48L2Rpdj5cbiJdfQ==
397
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2ViY2FtLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC1zY2FuZG9jL3NyYy9jb3JlL2NvbXBvbmVudHMvd2ViY2FtL3dlYmNhbS93ZWJjYW0uY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LXNjYW5kb2Mvc3JjL2NvcmUvY29tcG9uZW50cy93ZWJjYW0vd2ViY2FtL3dlYmNhbS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBRUwsdUJBQXVCLEVBRXZCLFNBQVMsRUFDVCxZQUFZLEVBQ1osWUFBWSxFQUNaLEtBQUssRUFHTCxNQUFNLEVBQ04sU0FBUyxHQUNWLE1BQU0sZUFBZSxDQUFDO0FBRXZCLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUNyRCxPQUFPLEVBQUUsSUFBSSxFQUF5QyxNQUFNLE1BQU0sQ0FBQztBQVluRSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sc0JBQXNCLENBQUM7Ozs7OztBQVVyRCxNQUFNLE9BQU8sZUFDWCxTQUFRLGFBQWE7SUF1RHJCLFlBQ1Msa0JBQXNDLEVBQ3RDLFFBQWtCLEVBQ2pCLEVBQXFCLEVBQ3JCLElBQVksRUFFWixjQUF3QztRQUVoRCxLQUFLLEVBQUUsQ0FBQztRQVBELHVCQUFrQixHQUFsQixrQkFBa0IsQ0FBb0I7UUFDdEMsYUFBUSxHQUFSLFFBQVEsQ0FBVTtRQUNqQixPQUFFLEdBQUYsRUFBRSxDQUFtQjtRQUNyQixTQUFJLEdBQUosSUFBSSxDQUFRO1FBRVosbUJBQWMsR0FBZCxjQUFjLENBQTBCO1FBekRsRCxlQUFVLEdBQVEsRUFBRSxLQUFLLEVBQUUsQ0FBQyxFQUFFLE1BQU0sRUFBRSxDQUFDLEVBQUUsQ0FBQztRQUMxQyxjQUFTLEdBQUcsS0FBSyxDQUFDO1FBRWxCLGVBQVUsR0FBUTtZQUNoQixHQUFH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QVM7dUJBQUMsUUFBUSxFQUFFLEVBQUUsTUFBTSxFQUFFLElBQUksRUFBRTtnQkFFUyxRQUFRO3NCQUFyRCxTQUFTO3VCQUFDLE9BQU8sRUFBRSxFQUFFLE1BQU0sRUFBRSxJQUFJLEVBQUU7Z0JBRW5CLGNBQWM7c0JBQTlCLE1BQU07Z0JBRVUsVUFBVTtzQkFBMUIsTUFBTTtnQkFFVSxZQUFZO3NCQUE1QixNQUFNO2dCQUdVLFNBQVM7c0JBQXpCLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBZnRlclZpZXdJbml0LFxuICBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSxcbiAgQ2hhbmdlRGV0ZWN0b3JSZWYsXG4gIENvbXBvbmVudCxcbiAgRXZlbnRFbWl0dGVyLFxuICBIb3N0TGlzdGVuZXIsXG4gIElucHV0LFxuICBOZ1pvbmUsXG4gIE9uRGVzdHJveSxcbiAgT3V0cHV0LFxuICBWaWV3Q2hpbGQsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgV2ViY2FtSW5pdEVycm9yIH0gZnJvbSAnLi4vZG9tYWluL3dlYmNhbS1pbml0LWVycm9yJztcbmltcG9ydCB7IFdlYmNhbUltYWdlIH0gZnJvbSAnLi4vZG9tYWluL3dlYmNhbS1pbWFnZSc7XG5pbXBvcnQgeyBmcm9tLCBPYnNlcnZhYmxlLCBvZiwgU3ViamVjdCwgU3Vic2NyaXB0aW9uIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBXZWJjYW1VdGlsIH0gZnJvbSAnLi4vdXRpbC93ZWJjYW0udXRpbCc7XG5pbXBvcnQgeyBXZWJjYW1NaXJyb3JQcm9wZXJ0aWVzIH0gZnJvbSAnLi4vZG9tYWluL3dlYmNhbS1taXJyb3ItcHJvcGVydGllcyc7XG5pbXBvcnQge1xuICBCcmVha3BvaW50T2JzZXJ2ZXIsXG4gIEJyZWFrcG9pbnRzLFxuICBCcmVha3BvaW50U3RhdGUsXG59IGZyb20gJ0Bhbmd1bGFyL2Nkay9sYXlvdXQnO1xuaW1wb3J0IHsgUGxhdGZvcm0gfSBmcm9tICdAYW5ndWxhci9jZGsvcGxhdGZvcm0nO1xuaW1wb3J0IHsgc3dpdGNoTWFwIH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHsgd29ya2VydGV4dCB9IGZyb20gJy4uLy4uLy4uL2FwcC53b3JrZXInO1xuaW1wb3J0IHsgTmd4U2NhbmRvY0NhbWVyYVByb3ZpZGVyIH0gZnJvbSAnLi4vLi4vLi4vLi4vcHJvdmlkZXJzL2NhbWVyYS5wcm92aWRlcic7XG5pbXBvcnQgeyBCYXNlQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYmFzZS5jb21wb25lbnQnO1xuXG5kZWNsYXJlIHZhciBNZWRpYVN0cmVhbVRyYWNrUHJvY2Vzc29yOiBhbnk7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ25neC1zY2FuZG9jLXdlYmNhbScsXG4gIHRlbXBsYXRlVXJsOiAnLi93ZWJjYW0uY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi93ZWJjYW0uY29tcG9uZW50LnNjc3MnXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG59KVxuZXhwb3J0IGNsYXNzIFdlYmNhbUNvbXBvbmVudFxuICBleHRlbmRzIEJhc2VDb21wb25lbnRcbiAgaW1wbGVtZW50cyBBZnRlclZpZXdJbml0LCBPbkRlc3Ryb3lcbntcbiAgcHJpdmF0ZSB0cmlnZ2VyU3Vic2NyaXB0aW9uPzogU3Vic2NyaXB0aW9uO1xuICBjYW52YXNTaXplOiBhbnkgPSB7IHdpZHRoOiAwLCBoZWlnaHQ6IDAgfTtcbiAgc2hvd1ZpZGVvID0gZmFsc2U7XG5cbiAgY2FudmFzRGF0YTogYW55ID0ge1xuICAgIGN0eDogbnVsbCxcbiAgICBjdHhPOiBudWxsLFxuICB9O1xuXG4gIG1lZGlhU3RyZWFtITogTWVkaWFTdHJlYW07XG4gIEBJbnB1dCgpIGltYWdlSGFuZGxlciE6IFN1YmplY3Q8YW55PjtcblxuICBASW5wdXQoKSBwdWJsaWMgdHlwZT86ICdkb2N1bWVudCcgfCAnc2VsZmllJyB8ICdtYW51YWwnO1xuXG4gIEBJbnB1dCgpIHRyaWdnZXIhOiBPYnNlcnZhYmxlPG51bWJlcj47XG4gIC8vIHB1YmxpYyBzZXQgdHJpZ2dlcih0cmlnZ2VyOiBPYnNlcnZhYmxlPG51bWJlcj4pIHtcbiAgLy8gICBpZiAodGhpcy50cmlnZ2VyU3Vic2NyaXB0aW9uKSB7XG4gIC8vICAgICB0aGlzLnRyaWdnZXJTdWJzY3JpcHRpb24udW5zdWJzY3JpYmUoKTtcbiAgLy8gICB9XG5cbiAgLy8gICAvLyBTdWJzY3JpYmUgdG8gZXZlbnRzIGZyb20gdGhpcyBPYnNlcnZhYmxlIHRvIHRha2Ugc25hcHNob3RzXG4gIC8vICAgdGhpcy50cmlnZ2VyU3Vic2NyaXB0aW9uID0gdHJpZ2dlci5zdWJzY3JpYmUoKHRpbWUpID0+IHtcbiAgLy8gICAgIHRoaXMudGFrZVNuYXBzaG90KHRpbWUpO1xuICAvLyAgIH0pO1xuICAvLyB9XG4gIHJlc2l6ZVRpbWVvdXQ6IGFueTtcbiAgQEhvc3RMaXN0ZW5lcignd2luZG93OnJlc2l6ZScsIFsnJGV2ZW50J10pXG4gIG9uUmVzaXplKCkge1xuICAgIHRoaXMuY2FtZXJhUHJvdmlkZXIucmVjdFBvc2l0aW9uLnQgPSAwO1xuICAgIC8vIHRoaXMudmlkZW9SZWFkeS5uZXh0KGZhbHNlKTtcbiAgICBpZiAodGhpcy5yZXNpemVUaW1lb3V0KSB7XG4gICAgICBjbGVhclRpbWVvdXQodGhpcy5yZXNpemVUaW1lb3V0KTtcbiAgICB9XG4gICAgdGhpcy5yZXNpemVUaW1lb3V0ID0gc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICB0aGlzLmRyYXdSZWN0YW5nbGUoKTtcbiAgICB9LCA1MCk7XG4gIH1cblxuICBwcml2YXRlIGNodW5rczphbnkgPSBbXTtcbiAgQFZpZXdDaGlsZCgnY2FudmFzJywgeyBzdGF0aWM6IHRydWUgfSkgcHJpdmF0ZSBjYW52YXM6IGFueTtcblxuICBAVmlld0NoaWxkKCd2aWRlbycsIHsgc3RhdGljOiB0cnVlIH0pIHByaXZhdGUgdmlkZW9SZWY6IGFueTtcblxuICBAT3V0cHV0KCkgcHVibGljIGNhbWVyYVN3aXRjaGVkOiBFdmVudEVtaXR0ZXI8c3RyaW5nPiA9XG4gICAgbmV3IEV2ZW50RW1pdHRlcjxzdHJpbmc+KCk7XG4gIEBPdXRwdXQoKSBwdWJsaWMgdmlkZW9SZWFkeTogRXZlbnRFbWl0dGVyPGJvb2xlYW4+ID1cbiAgICBuZXcgRXZlbnRFbWl0dGVyPGJvb2xlYW4+KCk7XG4gIEBPdXRwdXQoKSBwdWJsaWMgaW1hZ2VDYXB0dXJlOiBFdmVudEVtaXR0ZXI8V2ViY2FtSW1hZ2U+ID1cbiAgICBuZXcgRXZlbnRFbWl0dGVyPFdlYmNhbUltYWdlPigpO1xuXG4gIEBPdXRwdXQoKSBwdWJsaWMgaW5pdEVycm9yOiBFdmVudEVtaXR0ZXI8V2ViY2FtSW5pdEVycm9yPiA9XG4gICAgbmV3IEV2ZW50RW1pdHRlcjxXZWJjYW1Jbml0RXJyb3I+KCk7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyBicmVha3BvaW50T2JzZXJ2ZXI6IEJyZWFrcG9pbnRPYnNlcnZlcixcbiAgICBwdWJsaWMgcGxhdGZvcm06IFBsYXRmb3JtLFxuICAgIHByaXZhdGUgY2Q6IENoYW5nZURldGVjdG9yUmVmLFxuICAgIHByaXZhdGUgem9uZTogTmdab25lLFxuXG4gICAgcHJpdmF0ZSBjYW1lcmFQcm92aWRlcjogTmd4U2NhbmRvY0NhbWVyYVByb3ZpZGVyXG4gICkge1xuICAgIHN1cGVyKCk7XG4gIH1cbiAgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIGNvbnNvbGUud2FybignW3ZpZGVvXSBzdGFydCcpO1xuICAgIC8vIGluaXRpYWwgbG9hZFxuICAgIHRoaXMuX19zdWJzKHRoaXMuY2FtZXJhUHJvdmlkZXIuZ2V0RGV2aWNlcygpKS5zdWJzY3JpYmUoKHJlc3ApID0+IHtcbiAgICAgIHRoaXMuaW5pdCgpO1xuICAgIH0pO1xuXG4gICAgLy8gY2FtZXJhIHN3aXRjaFxuICAgIHRoaXMuX19zdWJzKHRoaXMuY2FtZXJhUHJvdmlkZXIud2ViY2FtT2JzZXJ2YWJsZSkuc3Vic2NyaWJlKChkZXZpY2UpID0+IHtcbiAgICAgIGNvbnN0IHZpZGVvVHJhY2tDb25zdHJhaW50czogYW55ID0gdGhpcy5zZXRDb25zdHJhaW50cygpO1xuICAgICAgdGhpcy5zdG9wQWxsVHJhY2tzKCk7XG4gICAgICB0aGlzLnNob3dWaWRlbyA9IGZhbHNlO1xuICAgICAgdGhpcy5jZC5kZXRlY3RDaGFuZ2VzKCk7XG4gICAgICB2aWRlb1RyYWNrQ29uc3RyYWludHMuZGV2aWNlSWQgPSBkZXZpY2U7XG4gICAgICB0aGlzLnNldHVwKHZpZGVvVHJhY2tDb25zdHJhaW50cyk7XG4gICAgfSk7XG5cbiAgICB0aGlzLl9fc3Vicyh0aGlzLnRyaWdnZXIpLnN1YnNjcmliZSgodGltZSkgPT4gdGhpcy50YWtlU25hcHNob3QodGltZSkpO1xuICB9XG5cbiAgcHVibGljIGdldCB2aWRlbygpIHtcbiAgICByZXR1cm4gdGhpcy52aWRlb1JlZi5uYXRpdmVFbGVtZW50O1xuICB9XG4gIGdldCBpc01vYmlsZSgpIHtcbiAgICByZXR1cm4gdGhpcy5wbGF0Zm9ybS5BTkRST0lEIHx8IHRoaXMucGxhdGZvcm0uSU9TO1xuICB9XG4gIHByaXZhdGUgc2V0Q29uc3RyYWludHMoKSB7XG4gICAgY29uc3Qgb3V0OiBNZWRpYVN0cmVhbUNvbnN0cmFpbnRzID0ge307XG4gICAgc3dpdGNoICh0aGlzLnR5cGUpIHtcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIGlmICh0aGlzLmlzTW9iaWxlKSB7XG4gICAgICAgICAgb3V0LnZpZGVvID0geyBoZWlnaHQ6IHsgaWRlYWw6IDE2MDAgfSwgd2lkdGg6IHsgaWRlYWw6IDE2MDAgfSB9O1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIG91dC52aWRlbyA9IHsgaGVpZ2h0OiB7IGlkZWFsOiAxMDgwIH0sIHdpZHRoOiB7IGlkZWFsOiAxOTIwIH0gfTtcbiAgICAgICAgfVxuICAgICAgICBvdXQudmlkZW8uZmFjaW5nTW9kZSA9ICdlbnZpcm9ubWVudCc7XG4gICAgICAgIGJyZWFrO1xuXG4gICAgICBjYXNlICdzZWxmaWUnOlxuICAgICAgICBpZiAodGhpcy5pc01vYmlsZSkge1xuICAgICAgICAgIG91dC52aWRlbyA9IHsgaGVpZ2h0OiB7IGlkZWFsOiAxNjAwIH0sIHdpZHRoOiB7IGlkZWFsOiAxNjAwIH0gfTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBvdXQudmlkZW8gPSB7IGhlaWdodDogeyBpZGVhbDogMTA4MCB9LCB3aWR0aDogeyBpZGVhbDogMTkyMCB9IH07XG4gICAgICAgIH1cbiAgICAgICAgb3V0LnZpZGVvLmZhY2luZ01vZGUgPSAndXNlcic7XG4gICAgICAgIGJyZWFrO1xuICAgIH1cblxuICAgIHJldHVybiBvdXQudmlkZW87XG4gIH1cblxuICBwcml2YXRlIGdldCBhY3RpdmVUcmFja1NldHRpbmdzKCkge1xuICAgIHJldHVybiB0aGlzLm1lZGlhU3RyZWFtLmdldFZpZGVvVHJhY2tzKClbMF0uZ2V0U2V0dGluZ3MoKTtcbiAgfVxuICBwcml2YXRlIGdldCBhY3RpdmVUcmFja0NhcGFiaWxpdGllcygpIHtcbiAgICB0cnkge1xuICAgICAgcmV0dXJuIHRoaXMubWVkaWFTdHJlYW0uZ2V0VmlkZW9UcmFja3MoKVswXS5nZXRDYXBhYmlsaXRpZXMoKTtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICByZXR1cm4ge307XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBpbml0KCkge1xuICAgIGNvbnN0IHZpZGVvVHJhY2tDb25zdHJhaW50cyA9IHRoaXMuc2V0Q29uc3RyYWludHMoKTtcblxuICAgIHRoaXMuc2V0dXAodmlkZW9UcmFja0NvbnN0cmFpbnRzKTtcbiAgfVxuICBwcml2YXRlIHNldHVwKHZpZGVvVHJhY2tDb25zdHJhaW50czogYW55KSB7XG4gICAgdGhpcy5fX3N1YnMoXG4gICAgICBmcm9tKFxuICAgICAgICBuYXZpZ2F0b3IubWVkaWFEZXZpY2VzLmdldFVzZXJNZWRpYSg8TWVkaWFTdHJlYW1Db25zdHJhaW50cz57XG4gICAgICAgICAgdmlkZW86IHZpZGVvVHJhY2tDb25zdHJhaW50cyxcbiAgICAgICAgfSlcbiAgICAgIClcbiAgICApLnN1YnNjcmliZSgoc3RyZWFtOiBNZWRpYVN0cmVhbSkgPT4ge1xuICAgICAgdGhpcy5tZWRpYVN0cmVhbSA9IHN0cmVhbTtcbiAgICAgIHRoaXMuY2FtZXJhUHJvdmlkZXIuZGV2aWNlSWQgPSB0aGlzLmFjdGl2ZVRyYWNrU2V0dGluZ3MuZGV2aWNlSWQ7XG4gICAgICB0aGlzLnN0YXJ0VmlkZW8oKTtcbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhcnRWaWRlbygpIHtcbiAgICBjb25zb2xlLmxvZygnU1RBUlQgVklERU8nKTtcbiAgICB0aGlzLnZpZGVvLnNyY09iamVjdCA9IHRoaXMubWVkaWFTdHJlYW07XG5cbiAgICB0aGlzLnZpZGVvLm9ubG9hZGVkbWV0YWRhdGEgPSAoZGF0YTogYW55KSA9PiB7XG4gICAgICAvLyAgY29uc29sZS5sb2coZGF0YSk7XG4gICAgICB0aGlzLnZpZGVvLnBsYXkoKTtcbiAgICAgIHRoaXMuc2hvd1ZpZGVvID0gdHJ1ZTtcbiAgICAgIHRoaXMuY2QuZGV0ZWN0Q2hhbmdlcygpO1xuICAgICAgY29uc29sZS5sb2coJ1ZJREVPIFBMQVknKTtcbiAgICB9O1xuICAgIHRoaXMudmlkZW8ub25wbGF5ID0gKCkgPT4ge1xuICAgICAgdGhpcy52aWRlb1JlYWR5Lm5leHQodHJ1ZSk7XG4gICAgICBjb25zb2xlLmxvZyh0aGlzLnZpZGVvLnZpZGVvV2lkdGgsIHRoaXMudmlkZW8udmlkZW9IZWlnaHQpO1xuICAgICAgdGhpcy5kcmF3UmVjdGFuZ2xlKCk7XG4gICAgfTtcbiAgfVxuICBzZXRDYW52YXMoKSB7XG4gICAgbGV0IF9jYW52YXM6IEhUTUxDYW52YXNFbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnY2FudmFzJyk7IC8vdGhpcy5jYW52YXNTbmFwc2hvdC5uYXRpdmVFbGVtZW50O1xuXG4gICAgY29uc3QgeyB2aWRlb1dpZHRoLCB2aWRlb0hlaWdodCB9ID0gdGhpcy52aWRlbztcblxuICAgIGNvbnN0IGFzcGVjdCA9IHZpZGVvV2lkdGggLyB2aWRlb0hlaWdodDtcblxuICAgIGNvbnN0IGNhbnZhc1NtYWxTaXplID0gMzg0O1xuICAgIGNvbnN0IHNtYWxsU2l6ZSA9IHtcbiAgICAgIHc6IDAsXG4gICAgICBoOiAwLFxuICAgIH07XG5cbiAgICBpZiAodmlkZW9XaWR0aCA+PSB2aWRlb0hlaWdodCkge1xuICAgICAgc21hbGxTaXplLncgPSBjYW52YXNTbWFsU2l6ZTtcbiAgICAgIHNtYWxsU2l6ZS5oID0gY2FudmFzU21hbFNpemUgLyBhc3BlY3Q7XG4gICAgfSBlbHNlIHtcbiAgICAgIHNtYWxsU2l6ZS5oID0gY2FudmFzU21hbFNpemU7XG4gICAgICBzbWFsbFNpemUuaCA9IGNhbnZhc1NtYWxTaXplICogYXNwZWN0O1xuICAgIH1cblxuICAgIGNvbnNvbGUubG9nKCdTWicsIHNtYWxsU2l6ZSk7XG5cbiAgICBfY2FudmFzLndpZHRoID0gdmlkZW9XaWR0aDtcbiAgICBfY2FudmFzLmhlaWdodCA9IHZpZGVvSGVpZ2h0O1xuICAgIGNvbnN0IGN0eDogYW55ID0gX2NhbnZhcy5nZXRDb250ZXh0KCcyZCcsIHtcbiAgICAgIGFscGhhOiBmYWxzZSxcbiAgICAgIHBvd2VyUHJlZmVyZW5jZTogJ2hpZ2gtcGVyZm9ybWFuY2UnLFxuICAgICAgYW50aWFsaWFzOiBmYWxzZSxcbiAgICAgIGRlcHRoOiBmYWxzZSxcbiAgICAgIGRlc3luY2hyb25pemVkOiB0cnVlLFxuICAgIH0pO1xuXG4gICAgY29uc3QgY2FudmFzID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnY2FudmFzJyk7IC8vIG5lZWRzIGFuIGluaXRpYWwgc2l6ZVxuICAgIGNhbnZhcy5oZWlnaHQgPSBzbWFsbFNpemUuaDtcbiAgICBjYW52YXMud2lkdGggPSBzbWFsbFNpemUudztcblxuICAgIGNvbnN0IGN0eE86IGFueSA9IGNhbnZhcy5nZXRDb250ZXh0KCcyZCcsIHtcbiAgICAgIGFscGhhOiBmYWxzZSxcbiAgICAgIHBvd2VyUHJlZmVyZW5jZTogJ2hpZ2gtcGVyZm9ybWFuY2UnLFxuICAgICAgYW50aWFsaWFzOiBmYWxzZSxcbiAgICAgIGRlcHRoOiBmYWxzZSxcbiAgICAgIGRlc3luY2hyb25pemVkOiB0cnVlLFxuICAgIH0pO1xuXG4gICAgdGhpcy5jYW52YXNEYXRhLmN0eCA9IGN0eDtcbiAgICB0aGlzLmNhbnZhc0RhdGEuY3R4TyA9IGN0eE87XG4gICAgdGhpcy5jYW52YXNEYXRhLmNhbnZhcyA9IGNhbnZhcztcbiAgICB0aGlzLmNhbnZhc0RhdGEuX2NhbnZhcyA9IF9jYW52YXM7XG4gICAgdGhpcy5jYW52YXNEYXRhLnNpemUgPSBzbWFsbFNpemU7XG5cbiAgICAvLyBjb25zdCBzdHJlYW0gPSB0aGlzLnZpZGVvLmNhcHR1cmVTdHJlYW0oNSk7XG5cblxuXG4gICAgLy8gY29uc3QgbWVkaWFSZWNvcmRlciA9IG5ldyBNZWRpYVJlY29yZGVyKHN0cmVhbSk7XG5cblxuICAgIC8vIG1lZGlhUmVjb3JkZXIuc3RhcnQoKTtcblxuICAgIC8vIG1lZGlhUmVjb3JkZXIub25kYXRhYXZhaWxhYmxlID0gKGU6YW55KSA9PiB7XG4gICAgLy8gICBjb25zb2xlLmxvZyhlKVxuICAgIC8vICAgdGhpcy5jaHVua3MucHVzaChlLmRhdGEpO1xuXG4gICAgLy8gICB2YXIgdXJsID0gd2luZG93LlVSTC5jcmVhdGVPYmplY3RVUkwoZS5kYXRhKTtcbiAgICAvLyAgIHZhciBhbmNob3IgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdhJyk7XG4gICAgLy8gICBhbmNob3IuZG93bmxvYWQgPSAnbXlmaWxlLndlYm0nO1xuICAgIC8vICAgYW5jaG9yLmhyZWYgPSB1cmw7XG4gICAgLy8gICBhbmNob3IuY2xpY2soKTtcbiAgICAvLyB9O1xuICB9XG5cbiAgcHJpdmF0ZSB0YWtlU2VsZmllKCkge1xuICAgIGNvbnN0IG1pbWVUeXBlOiBzdHJpbmcgPSAnaW1hZ2UvanBlZyc7XG4gICAgY29uc3QgeyB2aWRlb1dpZHRoLCBvZmZzZXRXaWR0aCwgdmlkZW9IZWlnaHQgfSA9IHRoaXMudmlkZW87XG5cbiAgICBsZXQgX2NhbnZhczogSFRNTENhbnZhc0VsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdjYW52YXMnKTsgLy90aGlzLmNhbnZhc1NuYXBzaG90Lm5hdGl2ZUVsZW1lbnQ7XG5cbiAgICBjb25zdCB7IHcsIGggfSA9IHRoaXMuY2FtZXJhUHJvdmlkZXIucmVjdFBvc2l0aW9uO1xuXG4gICAgY29uc3QgYXNwZWN0ID0gdmlkZW9XaWR0aCAvIG9mZnNldFdpZHRoO1xuXG4gICAgLy8gY29uc29sZS5sb2codmlkZW9XaWR0aCwgb2Zmc2V0V2lkdGgsIGwsIHQsIHcsIGgpO1xuXG4gICAgX2NhbnZhcy53aWR0aCA9IHcgKiBhc3BlY3Q7XG4gICAgX2NhbnZhcy5oZWlnaHQgPSBoICogYXNwZWN0O1xuXG4gICAgY29uc3QgZHggPSB2aWRlb1dpZHRoIC0gdyAqIGFzcGVjdDtcbiAgICBjb25zdCBkeSA9IHZpZGVvSGVpZ2h0IC0gaCAqIGFzcGVjdDtcbiAgICBjb25zdCB4ID0gZHggLyAyO1xuICAgIGNvbnN0IHkgPSBkeSAvIDI7XG5cbiAgICBjb25zdCBjdHg6IGFueSA9IF9jYW52YXMuZ2V0Q29udGV4dCgnMmQnLCB7XG4gICAgICBhbHBoYTogZmFsc2UsXG4gICAgICBwb3dlclByZWZlcmVuY2U6ICdoaWdoLXBlcmZvcm1hbmNlJyxcbiAgICAgIGFudGlhbGlhczogZmFsc2UsXG4gICAgICBkZXB0aDogZmFsc2UsXG4gICAgICBkZXN5bmNocm9uaXplZDogdHJ1ZSxcbiAgICB9KTtcblxuICAgIGlmIChjdHgpIHtcbiAgICAgIGN0eC5kcmF3SW1hZ2UoXG4gICAgICAgIHRoaXMudmlkZW8sXG4gICAgICAgIHgsXG4gICAgICAgIHksXG4gICAgICAgIHcgKiBhc3BlY3QsXG4gICAgICAgIGggKiBhc3BlY3QsXG4gICAgICAgIDAsXG4gICAgICAgIDAsXG4gICAgICAgIHcgKiBhc3BlY3QsXG4gICAgICAgIGggKiBhc3BlY3RcbiAgICAgICk7XG5cbiAgICAgIGNvbnN0IGltZ0FzVXJsOiBhbnkgPSBfY2FudmFzLnRvRGF0YVVSTChtaW1lVHlwZSwgMSk7XG5cbiAgICAgIHRoaXMuaW1hZ2VDYXB0dXJlLmVtaXQoXG4gICAgICAgIG5ldyBXZWJjYW1JbWFnZShpbWdBc1VybCwgbWltZVR5cGUsIG5ldyBJbWFnZURhdGEoMSwgMSksIG51bGwpXG4gICAgICApO1xuICAgIH1cbiAgfVxuICB0YWtlU25hcHNob3QodGltZTogbnVtYmVyKSB7XG4gICAgaWYgKHRoaXMudHlwZSAhPT0gJ2RvY3VtZW50Jykge1xuICAgICAgdGhpcy50YWtlU2VsZmllKCk7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGNvbnNvbGUud2FybignU05BUCcsIHRpbWUpO1xuICAgIGlmICghdGhpcy5zaG93VmlkZW8pIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy56b25lLnJ1bk91dHNpZGVBbmd1bGFyKCgpID0+IHtcbiAgICAgIGNvbnN0IGNhbnZhc1NtYWxTaXplID0gMzg0O1xuXG4gICAgICBjb25zdCBtaW1lVHlwZTogc3RyaW5nID0gJ2ltYWdlL2pwZWcnO1xuXG4gICAgICAvLyBsZXQgX2NhbnZhczogSFRNTENhbnZhc0VsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdjYW52YXMnKTsgLy90aGlzLmNhbnZhc1NuYXBzaG90Lm5hdGl2ZUVsZW1lbnQ7XG5cbiAgICAgIGNvbnN0IHsgdmlkZW9XaWR0aCwgdmlkZW9IZWlnaHQgfSA9IHRoaXMudmlkZW87XG5cbiAgICAgIC8vIF9jYW52YXMud2lkdGggPSB2aWRlb1dpZHRoO1xuICAgICAgLy8gX2NhbnZhcy5oZWlnaHQgPSB2aWRlb0hlaWdodDtcbiAgICAgIC8vIGNvbnNvbGUubG9nKHZpZGVvV2lkdGgsIHZpZGVvSGVpZ2h0KTtcblxuICAgICAgLy8gY29uc3QgY3R4OiBhbnkgPSBfY2FudmFzLmdldENvbnRleHQoJzJkJywge1xuICAgICAgLy8gICBhbHBoYTogZmFsc2UsXG4gICAgICAvLyAgIHBvd2VyUHJlZmVyZW5jZTogJ2hpZ2gtcGVyZm9ybWFuY2UnLFxuICAgICAgLy8gICBhbnRpYWxpYXM6IGZhbHNlLFxuICAgICAgLy8gICBkZXB0aDogZmFsc2UsXG4gICAgICAvLyAgIGRlc3luY2hyb25pemVkOiB0cnVlLFxuICAgICAgLy8gfSk7XG5cbiAgICAgIC8vIGNvbnN0IGNhbnZhcyA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ2NhbnZhcycpOyAvLyBuZWVkcyBhbiBpbml0aWFsIHNpemVcbiAgICAgIC8vIGNhbnZhcy5oZWlnaHQgPSBjYW52YXNTbWFsU2l6ZTtcbiAgICAgIC8vIGNhbnZhcy53aWR0aCA9IGNhbnZhc1NtYWxTaXplO1xuXG4gICAgICAvLyBjb25zdCBjdHhPOiBhbnkgPSBjYW52YXMuZ2V0Q29udGV4dCgnMmQnLCB7XG4gICAgICAvLyAgIGFscGhhOiBmYWxzZSxcbiAgICAgIC8vICAgcG93ZXJQcmVmZXJlbmNlOiAnaGlnaC1wZXJmb3JtYW5jZScsXG4gICAgICAvLyAgIGFudGlhbGlhczogZmFsc2UsXG4gICAgICAvLyAgIGRlcHRoOiBmYWxzZSxcbiAgICAgIC8vICAgZGVzeW5jaHJvbml6ZWQ6IHRydWUsXG4gICAgICAvLyB9KTtcblxuICAgICAgY29uc3QgeyBjdHgsIGN0eE8sIGNhbnZhcywgX2NhbnZhcywgc2l6ZSB9ID0gdGhpcy5jYW52YXNEYXRhO1xuXG4gICAgICBpZiAoY3R4ICYmIGN0eE8pIHtcbiAgICAgICAgY3R4LmltYWdlU21vb3RoaW5nRW5hYmxlZCA9IGZhbHNlO1xuICAgICAgICBjb25zdCBkcmF3U3RhcnQgPSBEYXRlLm5vdygpO1xuICAgICAgICBjdHguY2xlYXJSZWN0KDAsIDAsIHZpZGVvV2lkdGgsIHZpZGVvSGVpZ2h0KTtcbiAgICAgICAgY3R4Ty5jbGVhclJlY3QoMCwgMCwgc2l6ZS53LCBzaXplLmgpO1xuXG4gICAgICAgIGN0eC5kcmF3SW1hZ2UodGhpcy52aWRlbywgMCwgMCk7XG4gICAgICAgIGNvbnNvbGUudGltZSgnZHJhd1ZpZGVvJyk7XG4gICAgICAgIGN0eE8uZHJhd0ltYWdlKF9jYW52YXMsIDAsIDAsIHNpemUudywgc2l6ZS5oKTtcbiAgICAgICAgY29uc29sZS50aW1lRW5kKCdkcmF3VmlkZW8nKTtcbiAgICAgICAgY29uc29sZS50aW1lKCdEUkFXJyk7XG5cbiAgICAgICAgY29uc3QgaW1EYXRhOiBhbnkgPSBjdHguZ2V0SW1hZ2VEYXRhKDAsIDAsIHZpZGVvV2lkdGgsIHZpZGVvSGVpZ2h0KTtcblxuICAgICAgICBjb25zb2xlLnRpbWVFbmQoJ0RSQVcnKTtcblxuICAgICAgICBjb25zb2xlLnRpbWUoJ0NUWCcpO1xuXG4gICAgICAgIGNvbnN0IHJlc2l6ZSA9IGNhbnZhcy50b0RhdGFVUkwobWltZVR5cGUsIDEpO1xuICAgICAgICBjb25zb2xlLnRpbWVFbmQoJ0NUWCcpO1xuICAgICAgICBjb25zdCBjdXJyZW50VGltZSA9IG5ldyBEYXRlKCkuZ2V0VGltZSgpO1xuICAgICAgICBjb25zdCBkaWZmID0gY3VycmVudFRpbWUgLSB0aW1lO1xuXG4gICAgICAgIC8vIHRvdGFsIHRpbWUgZm9yIHBhY2sgb25lIGltYWdlXG4gICAgICAgIGNvbnN0IGRyYXdUaW1lID0gRGF0ZS5ub3coKSAtIGRyYXdTdGFydDtcblxuICAgICAgICBjb25zb2xlLndhcm4oJ0RSQVcgVElNRScsIGRyYXdUaW1lKTtcbiAgICAgICAgY29uc3QgZGVsYXlNYXggPSBkcmF3VGltZSA8IDYwID8gMTAwIDogMjAwO1xuXG4gICAgICAgIGNvbnN0IGRlbGF5ID0gZGlmZiA+IGRlbGF5TWF4ID8gMCA6IGRlbGF5TWF4IC0gZGlmZjtcblxuICAgICAgICBjb25zb2xlLndhcm4odGhpcy5jaHVua3MpXG4gICAgICAgIGNvbnN0IHRpbWVvdXQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICB0aGlzLmltYWdlQ2FwdHVyZS5lbWl0KG5ldyBXZWJjYW1JbWFnZSgnJywgbWltZVR5cGUsIGltRGF0YSwgcmVzaXplKSk7XG4gICAgICAgICAgY2xlYXJUaW1lb3V0KHRpbWVvdXQpO1xuICAgICAgICB9LCBkZWxheSk7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIHN0b3BBbGxUcmFja3MoKSB7XG4gICAgdGhpcy5zaG93VmlkZW8gPSBmYWxzZTtcbiAgICBpZiAodGhpcy5tZWRpYVN0cmVhbSAmJiB0aGlzLm1lZGlhU3RyZWFtLmdldFRyYWNrcykge1xuICAgICAgLy8gZ2V0VHJhY2tzKCkgcmV0dXJucyBhbGwgbWVkaWEgdHJhY2tzICh2aWRlbythdWRpbylcblxuICAgICAgdGhpcy5tZWRpYVN0cmVhbS5nZXRUcmFja3MoKS5mb3JFYWNoKCh0cmFjazogTWVkaWFTdHJlYW1UcmFjaykgPT4ge1xuICAgICAgICB0cmFjay5zdG9wKCk7XG4gICAgICAgIHRoaXMudmlkZW8uc3JjT2JqZWN0LnJlbW92ZVRyYWNrKHRyYWNrKTtcbiAgICAgICAgdGhpcy5tZWRpYVN0cmVhbS5yZW1vdmVUcmFjayh0cmFjayk7XG4gICAgICB9KTtcbiAgICAgIHRoaXMudmlkZW8uc3JjT2JqZWN0ID0gbnVsbDtcblxuICAgICAgdGhpcy52aWRlby5sb2FkKCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBkcmF3UmVjdGFuZ2xlKCkge1xuICAgIHRoaXMuem9uZS5ydW4oKCkgPT4ge1xuICAgICAgY29uc3QgX2NhbnZhcyA9IHRoaXMuY2FudmFzLm5hdGl2ZUVsZW1lbnQ7XG4gICAgICBsZXQgcGFkZGluZyA9IDEwO1xuICAgICAgaWYgKCF0aGlzLmlzTW9iaWxlICYmIF9jYW52YXMud2lkdGggIT09IF9jYW52YXMuaGVpZ2h0KSB7XG4gICAgICAgIHBhZGRpbmcgPSA0MDtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGN0eCA9IF9jYW52YXMuZ2V0Q29udGV4dCgnMmQnKTtcbiAgICAgIGNvbnNvbGUubG9nKGN0eCk7XG4gICAgICBjdHguY2xlYXJSZWN0KDAsIDAsIF9jYW52YXMud2lkdGgsIF9jYW52YXMuaGVpZ2h0KTtcblxuICAgICAgY29uc3QgeyB3aWR0aCwgaGVpZ2h0IH0gPSBfY2FudmFzO1xuXG4gICAgICBjb25zb2xlLmxvZyh3aWR0aCwgaGVpZ2h0KTtcblxuICAgICAgY29uc3QgeyB2aWRlb1dpZHRoLCB2aWRlb0hlaWdodCB9ID0gdGhpcy52aWRlbztcblxuICAgICAgY29uc3QgZG9jU2l6ZSA9IHRoaXMudHlwZSA9PT0gJ3NlbGZpZScgPyAxIDogODYgLyA1NTtcblxuICAgICAgY29uc3QgYXNwZWN0ID0gdmlkZW9XaWR0aCAvIHZpZGVvSGVpZ2h0O1xuICAgICAgbGV0IF9oZWlnaHQ7XG4gICAgICBsZXQgX3dpZHRoO1xuXG4gICAgICBpZiAodmlkZW9XaWR0aCA+PSB2aWRlb0hlaWdodCkge1xuICAgICAgICBfd2lkdGggPSB3aWR0aDtcblxuICAgICAgICBfaGVpZ2h0ID0gX3dpZHRoIC8gYXNwZWN0O1xuXG4gICAgICAgIGlmIChfaGVpZ2h0ID4gaGVpZ2h0KSB7XG4gICAgICAgICAgX2hlaWdodCA9IGhlaWdodDtcbiAgICAgICAgICBfd2lkdGggPSBhc3BlY3QgKiBfaGVpZ2h0O1xuICAgICAgICB9XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBfaGVpZ2h0ID0gaGVpZ2h0O1xuICAgICAgICBfd2lkdGggPSBhc3BlY3QgKiBfaGVpZ2h0O1xuICAgICAgICBpZiAoX3dpZHRoID4gd2lkdGgpIHtcbiAgICAgICAgICBfd2lkdGggPSB3aWR0aDtcbiAgICAgICAgICBfaGVpZ2h0ID0gX3dpZHRoIC8gYXNwZWN0O1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBsZXQgdywgaDtcblxuICAgICAgaWYgKF93aWR0aCA8PSBfaGVpZ2h0KSB7XG4gICAgICAgIHcgPSBfd2lkdGggLSAyICogcGFkZGluZztcbiAgICAgICAgaCA9IHcgLyBkb2NTaXplO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgaCA9IF9oZWlnaHQgLSAyICogcGFkZGluZztcbiAgICAgICAgdyA9IGRvY1NpemUgKiBoO1xuICAgICAgfVxuXG4gICAgICBoID0gTWF0aC5mbG9vcihoKTtcbiAgICAgIHcgPSBNYXRoLmZsb29yKHcpO1xuXG4gICAgICBjb25zb2xlLmxvZyhfd2lkdGgsIF9oZWlnaHQpO1xuXG4gICAgICBjb25zdCBsZWZ0ID0gTWF0aC5mbG9vcigod2lkdGggLSB3KSAvIDIpO1xuICAgICAgY29uc3QgdG9wID0gTWF0aC5mbG9vcigoaGVpZ2h0IC0gaCkgLyAyKTtcblxuICAgICAgY3R4LnN0cm9rZVN0eWxlID0gJ3JlZCc7XG5cbiAgICAgIC8vIGN0eC5zdHJva2VSZWN0KGxlZnQsIHRvcCwgX3dpZHRoLCBfaGVpZ2h0KTtcblxuICAgICAgY3R4LnNoYWRvd0NvbG9yID0gJ3JlZCc7XG4gICAgICBjdHguc2hhZG93Qmx1ciA9IDE1O1xuICAgICAgY3R4LmJlZ2luUGF0aCgpO1xuXG4gICAgICAvLyBEcmF3IHVzaW5nIDVweCBmb3IgYm9yZGVyIHJhZGl1cyBvbiBhbGwgc2lkZXNcbiAgICAgIC8vIHN0cm9rZSBpdCBidXQgbm8gZmlsbFxuICAgICAgY3R4LnJvdW5kUmVjdChsZWZ0LCB0b3AsIHcsIGgsIDUpO1xuICAgICAgY3R4LnN0cm9rZSgpO1xuXG4gICAgICB0aGlzLmNhbWVyYVByb3ZpZGVyLnJlY3RQb3NpdGlvbiA9IHtcbiAgICAgICAgbDogbGVmdCxcbiAgICAgICAgdDogdG9wLFxuICAgICAgICB3LFxuICAgICAgICBoLFxuICAgICAgICBfdzogX3dpZHRoLFxuICAgICAgICBfaDogX2hlaWdodCxcbiAgICAgIH07XG5cbiAgICAgIHRoaXMuc2V0Q2FudmFzKCk7XG4gICAgICAvLyBjdHguc3Ryb2tlKCk7XG4gICAgfSk7XG4gIH1cbiAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgdGhpcy52aWRlby5wYXVzZSgpO1xuXG4gICAgLy8gdGhpcy52aWRlby5nZXRUcmFja3MoKS5mb3JFYWNoKCh0cmFjazogYW55KSA9PiB7XG4gICAgLy8gICB0cmFjay5zdG9wKCk7XG4gICAgLy8gICB0aGlzLnZpZGVvLnNyY09iamVjdC5yZW1vdmVUcmFjayh0cmFjayk7XG4gICAgLy8gfSk7XG5cbiAgICB0aGlzLnN0b3BBbGxUcmFja3MoKTtcblxuICAgIGlmICh0aGlzLnRyaWdnZXJTdWJzY3JpcHRpb24pIHtcbiAgICAgIHRoaXMudHJpZ2dlclN1YnNjcmlwdGlvbi51bnN1YnNjcmliZSgpO1xuICAgIH1cbiAgICB0aGlzLl9fZGVzdHJveSgpO1xuXG4gICAgY29uc29sZS53YXJuKCdbVklERU9dIGRlc3Ryb3llZCcpO1xuICB9XG59XG4iLCI8ZGl2IGNsYXNzPVwid2ViY2FtLXdyYXBwZXJcIj5cbiAgPHZpZGVvICN2aWRlbyBhdXRvcGxheSBtdXRlZCBzdHlsZT1cImRpc3BsYXk6IGJsb2NrXCIgcGxheXNpbmxpbmU+PC92aWRlbz5cblxuICA8ZGl2IGNsYXNzPVwicmVjdGFuZ2xlXCI+XG4gICAgPGNhbnZhc1xuICAgICAgI2NhbnZhc1xuICAgICAgW25nU3R5bGVdPVwieyB2aXNpYmlsaXR5OiBzaG93VmlkZW8gPyAndmlzaWJsZScgOiAnaGlkZGVuJyB9XCJcbiAgICAgIFt3aWR0aF09XCJ2aWRlby5vZmZzZXRXaWR0aFwiXG4gICAgICBbaGVpZ2h0XT1cInZpZGVvLm9mZnNldEhlaWdodFwiXG4gICAgPjwvY2FudmFzPlxuICA8L2Rpdj5cbjwvZGl2PlxuIl19