annot-server/static/js/annotviz.js
changeset 108 082b64a5c699
child 109 8546e2181a73
equal deleted inserted replaced
107:6d41506f9482 108:082b64a5c699
       
     1 !function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var n;"undefined"!=typeof window?n=window:"undefined"!=typeof global?n=global:"undefined"!=typeof self&&(n=self),n.annotviz=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
       
     2 /**
       
     3  * scripts/main.js
       
     4  *
       
     5  * This is the starting point for your application.
       
     6  * Take a look at http://browserify.org/ for more info
       
     7  */
       
     8 
       
     9 'use strict';
       
    10 
       
    11 var doubleroll = require('./doubleroll');
       
    12 var annotsroll = require('./annotsroll');
       
    13 var annotstimeline = require('./annotstimeline');
       
    14 var annotsvizview = require('./annotsvizview');
       
    15 var stageview = require('./stageview');
       
    16 var wswrapper = require('./wswrapper');
       
    17 var logger = require('./logger');
       
    18 
       
    19 var _ = require('lodash');
       
    20 
       
    21 module.exports = _({})
       
    22     .extend(doubleroll)
       
    23     .extend(annotsroll)
       
    24     .extend(annotstimeline)
       
    25     .extend(annotsvizview)
       
    26     .extend(stageview)
       
    27     .extend(wswrapper)
       
    28     .extend(logger)
       
    29     .value();
       
    30 },{"./annotsroll":2,"./annotstimeline":3,"./annotsvizview":4,"./doubleroll":5,"./logger":6,"./stageview":8,"./wswrapper":10,"lodash":"lodash"}],2:[function(require,module,exports){
       
    31 /**
       
    32 * js/annotsRoll.js
       
    33 *
       
    34 * annotsRoll basic component
       
    35 *
       
    36 */
       
    37 
       
    38 'use strict';
       
    39 
       
    40 var PIXI = require('pixi');
       
    41 var _ = require('lodash');
       
    42 
       
    43 var DEFAULT_ANNOT_COLOR = '#bababa';
       
    44 
       
    45 var defaultAnnotStyles = {
       
    46     'label': { font: '16pt Arial Bold', fill: '#65A954', wordWrap: true},
       
    47     'text' : { font: '12pt Arial Regular', fill: '#444444', wordWrap: true},
       
    48     'user' : { font: '14pt Arial regular', fill: '#666666' },
       
    49 };
       
    50 
       
    51 var defaultOptions = {
       
    52     externalRefresh: false,
       
    53     defaultColor: DEFAULT_ANNOT_COLOR,
       
    54     annotStyles: defaultAnnotStyles
       
    55 };
       
    56 
       
    57 function AnnotsRoll(options) {
       
    58 
       
    59 //parentContainer, xInit, yInit, width, height, widthRoll, pixelsPerSecond, annotColors
       
    60     var _this = this;
       
    61     var opts = _(options).defaults(defaultOptions).value();
       
    62 
       
    63 
       
    64     this.container = new PIXI.DisplayObjectContainer();
       
    65     this.container.x = opts.xInit;
       
    66     this.container.y = opts.yInit;
       
    67     this.container.width = opts.width;
       
    68 
       
    69     this.height = opts.height;
       
    70     this.width = opts.width;
       
    71     this.widthRoll = opts.widthRoll;
       
    72     this.pixelsPerSecond = opts.pixelsPerSecond;
       
    73     this.annotColors = opts.annotColors;
       
    74     this.startTs = options.startTs || Date.now();
       
    75 
       
    76     var yInit = opts.yInit;
       
    77     var annotStyles = _(opts.annotStyles).defaults(defaultAnnotStyles).value();
       
    78     for(var style in annotStyles) {
       
    79     	if (annotStyles[style].wordWrap === true){
       
    80     		annotStyles[style].wordWrapWidth = this.widthRoll; 
       
    81     	}
       
    82     }
       
    83     console.log(annotStyles);
       
    84     var started = false;
       
    85     var ws = opts.ws;
       
    86     var externalRefresh = opts.externalRefresh;
       
    87     var stageView = opts.stageView;
       
    88     
       
    89     stageView.registerComponent(this);
       
    90 
       
    91     var isHidden = function(child) {
       
    92         // TODO: the origin point is an approximation. Should refine this
       
    93         var globalPos = child.toGlobal(new PIXI.Point(0,0));
       
    94         return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ;
       
    95     };
       
    96 
       
    97     this.addAnnots = function(data) {
       
    98 
       
    99         //var title = data.content.category.label;
       
   100         //var user = data.content.user;
       
   101         //Test cat and color
       
   102         //var colorAnnot = 0x65A954;
       
   103         var category = data.content.category.label,
       
   104             text     = data.content.text,
       
   105             user     = data.content.user,
       
   106             ts       = Date.parse(data.ts),
       
   107             color    = this.getColor(ts, data.content.category.code);
       
   108 
       
   109         this.addAnnot(category, text, user, color, ts);
       
   110     };
       
   111 
       
   112     this.getColor = function(ts, code) {
       
   113         var colorsDef;
       
   114         _(this.annotColors).eachRight(function(cdef) {
       
   115             console.log("cDef", cdef);
       
   116             console.log("cDef ts", cdef.ts, ts);
       
   117             if(cdef.ts < ts) {
       
   118                 colorsDef = cdef.colors;
       
   119                 return false;
       
   120             }
       
   121         });
       
   122         var resColor;
       
   123         console.log("colorsDef", colorsDef);
       
   124         if(colorsDef) {
       
   125             resColor = colorsDef[code];
       
   126         }
       
   127         if(!resColor) {
       
   128             resColor = DEFAULT_ANNOT_COLOR;
       
   129         }
       
   130         return resColor;
       
   131     }
       
   132 
       
   133     this.addAnnot = function(category, text, user, catColor, ts){
       
   134 
       
   135         var color = catColor ? catColor : DEFAULT_ANNOT_COLOR;
       
   136         var x = 0;
       
   137         var y = (ts-this.startTs) * this.pixelsPerSecond / 1000 + yInit;
       
   138 
       
   139         var colorHex = parseInt(color.replace(/^#/, ''), 16);
       
   140 
       
   141         var graphics = new PIXI.Graphics()
       
   142             .beginFill(colorHex)
       
   143             .drawRect(x, y, 10, 3)
       
   144             .endFill();
       
   145 
       
   146         this.container.addChild(graphics);
       
   147 
       
   148         var textHeight = 0;
       
   149         var catLabel = new PIXI.Text(
       
   150             category,
       
   151             _(annotStyles.label).extend({fill: color}).value()
       
   152         );
       
   153         catLabel.x = x + 20;
       
   154         catLabel.y = y - 23;
       
   155         this.container.addChild(catLabel);
       
   156         textHeight += (catLabel.height - 23 + 2);
       
   157 
       
   158         if(text) {
       
   159             var catText = new PIXI.Text(text, annotStyles.text);
       
   160             catText.x = x + 20;
       
   161             catText.y = y + 2;
       
   162             this.container.addChild(catText);
       
   163             textHeight += (catText.height + 2);
       
   164         }
       
   165 
       
   166         var catUser = new PIXI.Text(user, annotStyles.user);
       
   167         catUser.x = x + 20;
       
   168         catUser.y = y + 2 + textHeight;
       
   169         this.container.addChild(catUser);
       
   170 
       
   171         this.addAnnotLine(colorHex, y);
       
   172     };
       
   173 
       
   174     this.addAnnotLine = function(color, y) {
       
   175         var x = this.widthRoll;
       
   176 
       
   177 
       
   178         var graphics = new PIXI.Graphics()
       
   179             .beginFill(color)
       
   180             .drawRect(x, y, this.width - x, 3)
       
   181             .endFill();
       
   182 
       
   183         this.container.addChild(graphics);
       
   184     };
       
   185 
       
   186     this.moveTo = function(diffTime){
       
   187     	this.container.y = Math.floor(diffTime*this.pixelsPerSecond);
       
   188     };
       
   189 
       
   190     this.move = this.refresh = function() {
       
   191         var diff = (this.startTs - Date.now())/1000;
       
   192         this.moveTo(diff);
       
   193     };
       
   194 
       
   195     this.removePassedObjets = function(){
       
   196         var childrenToRemove = [];
       
   197         _(_this.container.children).forEach(function(child) {
       
   198             return typeof(child) === 'undefined' ||
       
   199                 (isHidden(child) && childrenToRemove.push(child));
       
   200         });
       
   201         childrenToRemove.forEach(function(child) {
       
   202             _this.container.removeChild(child);
       
   203         });
       
   204     };
       
   205 
       
   206     this.init = function() {
       
   207 
       
   208         ws.message(function(data) {
       
   209             _this.addAnnots(data);
       
   210         });
       
   211 
       
   212     };
       
   213 
       
   214     this.start = function() {
       
   215         if(!started) {
       
   216             this.startTs = Date.now();
       
   217             started = true;
       
   218         }
       
   219         this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.height / this.pixelsPerSecond );
       
   220         if(!externalRefresh) {
       
   221             this.refreshInterval = setInterval(function() {_this.move();}, 1000/this.framerate);
       
   222         }
       
   223     };
       
   224 
       
   225     this.stop = function() {
       
   226         clearInterval(this.cleanInterval);
       
   227         if(!externalRefresh) {
       
   228             clearInterval(this.refreshInterval);
       
   229         }
       
   230     };
       
   231 
       
   232 }
       
   233 
       
   234 module.exports = {
       
   235     AnnotsRoll: AnnotsRoll,
       
   236 };
       
   237 
       
   238 },{"lodash":"lodash","pixi":"pixi"}],3:[function(require,module,exports){
       
   239 /**
       
   240 * js/annotstimeline
       
   241 *
       
   242 * annotstimeline basic component
       
   243 *
       
   244 */
       
   245 
       
   246 'use strict';
       
   247 
       
   248 var PIXI = require('pixi');
       
   249 var Utils = require('./utils.js');
       
   250 var _ = require('lodash');
       
   251 
       
   252 var defaultOptions = {		
       
   253     logger: undefined,
       
   254     intervalWidth: 10,
       
   255     intervalHeight: 5,
       
   256     maxCellHeight: 200,
       
   257     radius: 300
       
   258 };
       
   259 
       
   260 
       
   261 function AnnotsTimeLine(options){
       
   262     var _this = this;
       
   263     var opts = _(options).defaults(defaultOptions).value();
       
   264     
       
   265     this.container = new PIXI.DisplayObjectContainer();
       
   266     this.container.x = opts.xInit;
       
   267     this.container.y = opts.yInit;
       
   268     this.container.width = opts.width;
       
   269     this.container.height = opts.height;    
       
   270     
       
   271     this.timeBegin = opts.timeBegin;
       
   272     this.timeEnd = opts.timeEnd;
       
   273     this.duration = (this.timeEnd - this.timeBegin)/1000;
       
   274     this.width = opts.width;
       
   275     this.height = opts.height;
       
   276     this.intervalHeight = opts.intervalHeight;
       
   277     this.intervalWidth = opts.intervalWidth;
       
   278     this.maxCellHeight = opts.maxCellHeight;
       
   279     this.annotCategories = opts.annotCategories;
       
   280     
       
   281     this.circleX = opts.circleX || (this.width/2);
       
   282     this.circleY = opts.circleY || (this.height/2);
       
   283     this.radius = opts.radius;
       
   284     this.perimeter = 2*Math.PI* this.radius;
       
   285     this.intervalDuration = (this.intervalWidth * this.duration / this.perimeter);
       
   286     
       
   287     var currentTime = this.timeBegin;
       
   288     var totalIndex = Math.floor(this.perimeter/this.intervalWidth);
       
   289     	
       
   290     this.cells = []
       
   291     for (var i=0; i<(this.perimeter/this.intervalWidth) ; i++){
       
   292     	this.cells[i] = [];
       
   293     	this.cells[i].i = i;
       
   294     	this.cells[i].totalAnnots = 0;
       
   295     	this.cells[i].categories = {};
       
   296     	
       
   297     	for (var category in this.annotCategories[0].colors){
       
   298     		this.cells[i].categories[category] = {
       
   299 				"count": 0,
       
   300 				"color": this.annotCategories[0].colors[category]
       
   301     		};
       
   302     	}
       
   303     }
       
   304     
       
   305     var ws = opts.ws;
       
   306     var stageView = opts.stageView;
       
   307 
       
   308     //draw the base - circle and line to locate the scene
       
   309     var graphics = new PIXI.Graphics();
       
   310     graphics.lineStyle(2, 0x646464)
       
   311     	.drawCircle(this.circleX, this.circleY, this.radius - 3)
       
   312     	.lineStyle(1, 0xD7D7D7)
       
   313     	.drawCircle(this.circleX, this.circleY, this.radius*2/3)
       
   314     	.drawCircle(this.circleX, this.circleY, this.radius/3)
       
   315     	.lineStyle(1, 0x646464)
       
   316     	.moveTo(this.circleX, this.circleY - (this.radius/3)/2)
       
   317     	.lineTo(this.circleX, this.circleY - this.radius - this.maxCellHeight - 10)
       
   318     	.endFill()
       
   319     this.container.addChild(graphics);
       
   320     
       
   321     //set time text
       
   322     var currentTimeText = new PIXI.Text("-- : -- : --", { font: '18pt Gothic Standard', fill: '#646464' });
       
   323     currentTimeText.x = this.circleX - currentTimeText.width/2;
       
   324     currentTimeText.y = this.circleY - currentTimeText.height/2;
       
   325     this.container.addChild(currentTimeText);
       
   326     
       
   327     stageView.registerComponent(this);
       
   328 
       
   329     //Add Annotation to the TimeLine
       
   330     this.addAnnot = function(data){
       
   331     	if (typeof(this.annotCategories[0].colors[data.content.category.code]) !== 'undefined'){
       
   332     		var annotCode = data.content.category.code;
       
   333     	} else {
       
   334     		var annotCode = this.annotCategories[0].order[this.annotCategories[0].order.length -1];
       
   335     	}
       
   336     	var annotTime = Date.parse(data.ts);
       
   337     	
       
   338     	if (this.timeEnd > Date.parse(data.ts)){
       
   339 	    	var i = Math.floor((Date.parse(data.ts)-this.timeBegin)/(1000*this.intervalDuration));
       
   340 	    	
       
   341 			this.cells[i].categories[annotCode].count += 1;
       
   342 			this.cells[i].totalAnnots +=1;
       
   343 			this.redrawCell(this.cells[i], i);
       
   344     	}
       
   345     };
       
   346     
       
   347     this.initGraphics = function(cell){
       
   348     	cell.graphics = new PIXI.Graphics();
       
   349     	cell.graphics.position.x = this.circleX + this.radius * Math.sin(cell.i*(360/totalIndex)*(Math.PI/180));
       
   350     	cell.graphics.position.y = this.circleY - this.radius * Math.cos(cell.i*(360/totalIndex)*(Math.PI/180));
       
   351     	cell.graphics.rotation = (cell.i)*(360/totalIndex)*(Math.PI/180) + (360/(totalIndex*2))*(Math.PI/180);
       
   352     	this.container.addChild(cell.graphics);
       
   353     }
       
   354     
       
   355     this.initTimeTexts = function() {
       
   356 	    var tBeg = new PIXI.Text(Utils.formatTime(this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
       
   357 	    tBeg.x = this.circleX + 15;
       
   358 	    tBeg.y = this.circleY - this.radius - this.maxCellHeight - 10;
       
   359 	    this.container.addChild(tBeg);
       
   360 	    
       
   361 	    var tEnd = new PIXI.Text(Utils.formatTime(this.timeEnd), { font: '12pt Gothic Standard', fill: '#646464' });
       
   362 	    tEnd.x = this.circleX - 15 - tEnd.width;
       
   363 	    tEnd.y = this.circleY - this.radius - this.maxCellHeight - 10;
       
   364 	    this.container.addChild(tEnd);
       
   365 	    
       
   366 	    var t15 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
       
   367 	    t15.x = this.circleX + this.radius + this.maxCellHeight + 10 ;
       
   368 	    t15.y = this.circleY - t15.height;
       
   369 	    t15.rotation = Math.PI /2;
       
   370 	    this.container.addChild(t15);
       
   371 	    
       
   372 	    var t30 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/2) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
       
   373 	    t30.x = this.circleX - t30.width/2;
       
   374 	    t30.y = this.circleY + this.radius + this.maxCellHeight - 2;
       
   375 	    this.container.addChild(t30);
       
   376 	    
       
   377 	    var t45 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)*3/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
       
   378 	    t45.x = this.circleX - this.radius - this.maxCellHeight - 10 ;
       
   379 	    t45.y = this.circleY + t15.height;
       
   380 	    t45.rotation = -Math.PI/2;
       
   381 	    this.container.addChild(t45);
       
   382     }
       
   383     
       
   384     //Draw the cellule
       
   385     this.redrawCell = function(cell){
       
   386     	
       
   387     	if (typeof(cell.graphics) === 'undefined'){
       
   388     		this.initGraphics(cell);
       
   389     	} else {
       
   390     		cell.graphics.clear();
       
   391     	}
       
   392     	
       
   393     	var y = 0;
       
   394     	
       
   395     	//Check if total height is higher than Max Cell Height
       
   396     	if ((cell.totalAnnots*this.intervalHeight) > this.maxCellHeight){
       
   397     		var heightStep = this.maxCellHeight/cell.totalAnnots;
       
   398     	} else {
       
   399     		var heightStep = this.intervalHeight;
       
   400     	}
       
   401     	
       
   402     	//Draw the rect depending on the height step calculated
       
   403     	for (var i=0; i< this.annotCategories[0].order.length; i++){
       
   404     		var currentCode = this.annotCategories[0].order[i];
       
   405 			cell.graphics.beginFill(cell.categories[currentCode].color.replace("#", "0x"))
       
   406     			.drawRect(0, y, this.intervalWidth-1, -cell.categories[currentCode].count * heightStep)
       
   407     			.endFill();
       
   408     		y -= cell.categories[currentCode].count*heightStep;
       
   409     	}
       
   410     }
       
   411     
       
   412     this.init = function() {
       
   413     	ws.message(function(data) {
       
   414             _this.addAnnot(data);
       
   415         });
       
   416     	
       
   417     	this.initTimeTexts();
       
   418     };
       
   419     
       
   420     this.updateTime = function(){
       
   421     	currentTime += 1000;
       
   422     	
       
   423         var nbSec = currentTime / 1000;
       
   424         var hours = Math.floor( nbSec / 3600 ) % 24;
       
   425         var minutes = Math.floor( nbSec / 60 ) % 60;
       
   426         var seconds = Math.floor(nbSec % 60);
       
   427         var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds  < 10 ? '0' + seconds : seconds);
       
   428         
       
   429         currentTimeText.setText(timeStr);
       
   430     };
       
   431     
       
   432     var refreshTimeInterval;
       
   433     
       
   434     this.start = function() {
       
   435     	refreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000);
       
   436     };
       
   437     
       
   438     this.refresh = function() {
       
   439     	
       
   440     };
       
   441     
       
   442     this.stop = function(){
       
   443     	console.log(this.cells);
       
   444     };
       
   445     
       
   446     return this;
       
   447 }
       
   448 
       
   449 module.exports = {
       
   450 	AnnotsTimeLine: AnnotsTimeLine
       
   451 };
       
   452 
       
   453 },{"./utils.js":9,"lodash":"lodash","pixi":"pixi"}],4:[function(require,module,exports){
       
   454 /**
       
   455 * js/annotsvizview.js
       
   456 *
       
   457 * This is the starting point for your application.
       
   458 * Take a look at http://browserify.org/ for more info
       
   459 */
       
   460 
       
   461 'use strict';
       
   462 
       
   463 var PIXI = require('pixi');
       
   464 var _ = require('lodash');
       
   465 var DoubleRoll = require('./doubleroll.js');
       
   466 var AnnotsTimeLine = require('./annotstimeline.js');
       
   467 var AnnotsRoll = require('./annotsroll.js');
       
   468 
       
   469 var defaultOptions = {
       
   470     xInit: 0,
       
   471     yInit: 0,
       
   472     width: 1024,
       
   473     height: 768,
       
   474     annotCategories: [{
       
   475         ts: 0,
       
   476         colors: {
       
   477             'ntm': '#CDC83F',
       
   478             'iam': '#CDC83F',
       
   479             'hip': '#CDC83F',
       
   480             'hop': '#CDC83F',
       
   481             'rock': '#DE8B53',
       
   482             'rap': '#DE8B53',
       
   483             'classic': '#DE8B53',
       
   484             'drums': '#C5A3CA',
       
   485             'guitar': '#C5A3CA',
       
   486             'bass': '#79BB92',
       
   487             'default': '#808080'
       
   488         },
       
   489         order: ['ntm', 'iam', 'hip', 'hop', 'rock', 'rap', 'classic', 'drums', 'guitar', 'bass', 'default']
       
   490     }]
       
   491 };
       
   492 
       
   493 function AnnotsVizView(options){
       
   494 	var _this = this;
       
   495     var opts = _(options).defaults(defaultOptions).value();
       
   496 
       
   497     this.container = new PIXI.DisplayObjectContainer();
       
   498     this.container.x = opts.xInit;
       
   499     this.container.y = opts.yInit;
       
   500 	this.width = opts.width;
       
   501 	this.height= opts.height;
       
   502     this.annotCategories = opts.annotCategories;
       
   503 
       
   504 	var wsPianoroll = opts.wsPianoroll;
       
   505 	var wsAnnot = opts.wsAnnot;
       
   506 	var stageView = opts.stageView;
       
   507 
       
   508 	stageView.registerComponent(this);
       
   509 
       
   510 	var timeLine = new AnnotsTimeLine.AnnotsTimeLine({
       
   511     	stageView : stageView,
       
   512         logger: logger,
       
   513         ws: new annotviz.WsWrapper(wsUriAnnotation, logger),
       
   514         xInit: 0,
       
   515         yInit: 0,
       
   516         width: 1024 - 200 - 200,
       
   517         height: 768-200,
       
   518         timeBegin: Date.now(),
       
   519         timeEnd: Date.now() + 3000000,
       
   520         intervalWidth: 6,
       
   521         intervalHeight: 10,
       
   522         maxCellHeight: 70,
       
   523         radius: 200,
       
   524         annotCategories: this.annotCategories
       
   525     });
       
   526 
       
   527 	var doubleRollH = new DoubleRoll.DoubleRoll({
       
   528         stageView : stageView,
       
   529     	logger: logger,
       
   530         ws: wsPianoroll,
       
   531         yInit: (this.height - 200),
       
   532         sceneHeight: 200,
       
   533         pianorolls : [
       
   534             {
       
   535                 height: 200,
       
   536                 timeWidth: 10,
       
   537                 lineInterval: 5000,
       
   538                 noteHeight: 10
       
   539             },
       
   540         ]
       
   541     });
       
   542 
       
   543 	var doubleRollV = new DoubleRoll.DoubleRoll({
       
   544         stageView : stageView,
       
   545     	logger: logger,
       
   546         ws: wsPianoroll,
       
   547         orientation: 'vertical',
       
   548         sceneHeight: 768-200,
       
   549         pianorolls : [
       
   550             {
       
   551                 height: 200,
       
   552                 timeWidth: 60,
       
   553                 lineInterval: 5000,
       
   554                 noteHeight: 5,
       
   555             },
       
   556         ]
       
   557     });
       
   558 
       
   559 	var annotsRoll = new AnnotsRoll.AnnotsRoll({
       
   560     	stageView : stageView,
       
   561         logger: logger,
       
   562         ws: wsAnnot,
       
   563         parentContainer: doubleRollV.stage,
       
   564         xInit: 1024 - 200 - 200,
       
   565         yInit: 768-200,
       
   566         width: 200 + 200,
       
   567         height: 768-200,
       
   568         widthRoll: 200,
       
   569         framerate: doubleRollV.framerate,
       
   570         pixelsPerSecond: Math.floor(1024 / 60),
       
   571         annotColors: this.annotCategories
       
   572     });
       
   573 
       
   574 	var limiters = new PIXI.Graphics()
       
   575 		.lineStyle(1, 0x646464)
       
   576 		.moveTo(annotsRoll.container.x, annotsRoll.container.y)
       
   577 		.lineTo(annotsRoll.container.x, annotsRoll.container.y - annotsRoll.height)
       
   578 		.moveTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y)
       
   579 		.lineTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y - annotsRoll.height)
       
   580 		.moveTo(0, this.height - 200)
       
   581 		.lineTo(this.width, this.height - 200)
       
   582 		.drawRect(0, 0, this.width -1, this.height -1)
       
   583 		.beginFill(0xECECEC)
       
   584 		.drawRect(1024 - 200, 0, 200, 768-200)
       
   585 		.endFill();
       
   586 	this.container.addChild(limiters);
       
   587 
       
   588 //	var doubleRollV = new DoubleRoll.DoubleRoll({});
       
   589 
       
   590 	this.init = function(){
       
   591 
       
   592 	}
       
   593 
       
   594 	this.start = function() {
       
   595     };
       
   596 
       
   597     this.refresh = function() {
       
   598     };
       
   599 
       
   600     this.stop = function(){
       
   601     };
       
   602 
       
   603     return this;
       
   604 
       
   605 }
       
   606 
       
   607 module.exports = {
       
   608 	AnnotsVizView: AnnotsVizView
       
   609 };
       
   610 
       
   611 },{"./annotsroll.js":2,"./annotstimeline.js":3,"./doubleroll.js":5,"lodash":"lodash","pixi":"pixi"}],5:[function(require,module,exports){
       
   612 /**
       
   613 * scripts/doubleroll.js
       
   614 *
       
   615 * This is the starting point for your application.
       
   616 * Take a look at http://browserify.org/ for more info
       
   617 */
       
   618 
       
   619 /* global document: false */
       
   620 
       
   621 'use strict';
       
   622 
       
   623 
       
   624 var PIXI = require('pixi');
       
   625 var _ = require('lodash');
       
   626 var PianoRoll = require('./pianoroll.js');
       
   627 
       
   628 var defaultConfig = {
       
   629     orientation: 'horizontal',
       
   630     logger: undefined,
       
   631     sceneWidth: 1024,
       
   632     pianorolls : [
       
   633       {
       
   634         height: 435,
       
   635         timeWidth: 10,
       
   636         lineInterval: 5000,
       
   637         noteHeight: undefined
       
   638       },
       
   639       {
       
   640         height: 645,
       
   641         timeWidth: 60,
       
   642         lineInterval: 5000,
       
   643         noteHeight: undefined
       
   644       },
       
   645     ],
       
   646     framerate: 25,
       
   647     offsetMusic: false,
       
   648     sceneBgColor: 0xFFFFFF,
       
   649     lineColor: 0x444444,
       
   650     lineFillColor: 0xFFFF00,
       
   651     noteColors: [0xB90000, 0x4BDD71, 0xAF931E, 0x1C28BA, 0x536991],
       
   652     noteHeight: undefined,
       
   653     zeroShift: 0.9,
       
   654     timeWidth: 60,
       
   655     lineInterval: 5000,
       
   656 //    wsUri: undefined,
       
   657 //    eventCode: undefined
       
   658 
       
   659 };
       
   660 
       
   661 function DoubleRoll(options) {
       
   662 
       
   663     var _this = this;
       
   664     var opts = _(options).defaults(defaultConfig).value();
       
   665 
       
   666     var orientation = opts.orientation;
       
   667     var isHorizontal = (orientation !== 'vertical');
       
   668 
       
   669     this.logger = opts.logger;
       
   670     this.lineColor = opts.lineColor;
       
   671     this.lineFillColor = opts.lineFillColor;
       
   672     this.framerate = opts.framerate;
       
   673     this.offsetMusic = opts.offsetMusic;
       
   674     this.noteColors = opts.noteColors;
       
   675 
       
   676     var noteHeight = opts.noteHeight;
       
   677     var sceneBgColor = opts.sceneBgColor;
       
   678     var sceneHeight = opts.sceneHeight || _(opts.pianorolls).reduce(function(s,p) { return s + p.height; }, 0);
       
   679     var timeWidth = opts.timeWidth;
       
   680     var lineInterval = opts.lineInterval;
       
   681     var offsetMusic = opts.offsetMusic;
       
   682 
       
   683     var sceneWidth = opts.sceneWidth;
       
   684     var stageView = opts.stageView;
       
   685 
       
   686     var zeroShift = opts.zeroShift;
       
   687 
       
   688     var ws = opts.ws;
       
   689 
       
   690     var colorsReg = {};
       
   691 
       
   692     this.container = new PIXI.DisplayObjectContainer();
       
   693     this.container.x = Math.floor(sceneWidth*zeroShift);
       
   694     this.container.y = 0;
       
   695     
       
   696     stageView.registerComponent(this);
       
   697 
       
   698     var pianorollList = [];
       
   699 
       
   700     var pianorollOptions = {
       
   701         parentContainer: this.container,
       
   702         orientation: orientation,
       
   703         xInit: 0,
       
   704         width: sceneWidth,
       
   705         noteColors: this.noteColors,
       
   706         colorsReg: colorsReg,
       
   707         lineColor: this.lineColor,
       
   708         lineInterval: lineInterval,
       
   709         offsetMusic: offsetMusic,
       
   710     };
       
   711 
       
   712     var yInit = opts.yInit || 0;
       
   713     var linesDown = true;
       
   714     _(opts.pianorolls).forEach(function(prDef, i) {
       
   715         var prNoteHeight = noteHeight || prDef.noteHeight || prDef.height / 128;
       
   716         var prTimeWidth = prDef.timeWidth || timeWidth;
       
   717         pianorollList.push(new PianoRoll(_({
       
   718             yInit: yInit,
       
   719             height: prDef.height,
       
   720             linesDown: linesDown,
       
   721             pixelsPerSecond: Math.floor(sceneWidth / prTimeWidth),
       
   722             noteHeight: prNoteHeight,
       
   723             lineInterval: prDef.lineInterval
       
   724         }).defaults(pianorollOptions).value()));
       
   725         yInit += prDef.height;
       
   726         linesDown = !linesDown;
       
   727 
       
   728         if(i<(opts.pianorolls.length-1)) {
       
   729             var lineGraphics = new PIXI.Graphics()
       
   730                 .beginFill(_this.lineFillColor)
       
   731                 .lineStyle(1, _this.lineColor)
       
   732                 .moveTo(Math.floor(sceneWidth*zeroShift), yInit)
       
   733                 .lineTo(-sceneWidth - Math.floor(sceneWidth*zeroShift), yInit)
       
   734                 .endFill();
       
   735             _this.container.addChild(lineGraphics);
       
   736         }
       
   737     });
       
   738 
       
   739     if(!isHorizontal) {
       
   740         this.container.rotation = Math.PI/2;
       
   741         this.container.y = sceneHeight;
       
   742         this.container.x = sceneWidth;
       
   743     }
       
   744 
       
   745 
       
   746     this.init = function() {
       
   747 
       
   748     	ws.message(function(data) {
       
   749             _this.addNotes(data);
       
   750         });
       
   751 
       
   752     };
       
   753 
       
   754 
       
   755     this.addNotes = function(data) {
       
   756 
       
   757         pianorollList.forEach(function(c) {
       
   758             c.addNoteRaw(data);
       
   759         });
       
   760     };
       
   761 
       
   762     this.refresh = function() {
       
   763         pianorollList.forEach(function(c) {
       
   764             c.move();
       
   765         });
       
   766     };
       
   767 
       
   768     // Init page and intervals
       
   769     var startTs;
       
   770 
       
   771     this.start = function() {
       
   772 
       
   773         startTs = Date.now();
       
   774         pianorollList.forEach(function(c) {
       
   775             c.start();
       
   776         });
       
   777     };
       
   778 
       
   779     this.stop = function() {
       
   780     	
       
   781         pianorollList.forEach(function(c) {
       
   782             c.stop();
       
   783         });
       
   784     };
       
   785 
       
   786 
       
   787     this.log = function(m) {
       
   788         if(this.logger) {
       
   789             this.logger.log(m);
       
   790         }
       
   791     };
       
   792 
       
   793 
       
   794 
       
   795     return this;
       
   796 }
       
   797 
       
   798 module.exports = {
       
   799     DoubleRoll: DoubleRoll
       
   800 };
       
   801 
       
   802 },{"./pianoroll.js":7,"lodash":"lodash","pixi":"pixi"}],6:[function(require,module,exports){
       
   803 /**
       
   804 * js/wswrapper.js
       
   805 *
       
   806 * simple logger service
       
   807 *
       
   808 */
       
   809 
       
   810 /* global document: false */
       
   811 
       
   812 'use strict';
       
   813 
       
   814 function HtmlLogger(doLog, container) {
       
   815 
       
   816     var logContainer = container;
       
   817     if(typeof(container) === 'string') {
       
   818         logContainer = document.getElementById(container);
       
   819     }
       
   820     if(!doLog) {
       
   821         document.body.removeChild(logContainer);
       
   822         logContainer = undefined;
       
   823     }
       
   824 
       
   825 
       
   826     this.log = function(msg) {
       
   827         if(doLog && logContainer) {
       
   828             logContainer.innerHTML += msg + '\n';
       
   829             logContainer.scrollTop = logContainer.scrollHeight;
       
   830         }
       
   831     };
       
   832 }
       
   833 
       
   834 function ConsoleLogger(doLog) {
       
   835 
       
   836     this.log = function(msg) {
       
   837         if(doLog) {
       
   838             console.log(msg);
       
   839         }
       
   840     }
       
   841 
       
   842 }
       
   843 
       
   844 module.exports = {
       
   845     HtmlLogger: HtmlLogger,
       
   846     ConsoleLogger: ConsoleLogger
       
   847 };
       
   848 
       
   849 },{}],7:[function(require,module,exports){
       
   850 /**
       
   851 * js/pianoroll.js
       
   852 *
       
   853 * pianoroll basic component
       
   854 *
       
   855 */
       
   856 
       
   857 'use strict';
       
   858 
       
   859 
       
   860 var PIXI = require('pixi');
       
   861 var randomColor = require('randomColor');
       
   862 var _ = require('lodash');
       
   863 
       
   864 var NTP_EPOCH_DELTA = 2208988800; //c.f. RFC 868
       
   865 
       
   866 function PianoRoll(options) {
       
   867     var _this = this;
       
   868     this.container = new PIXI.DisplayObjectContainer();
       
   869     this.container.x = options.xInit;
       
   870     this.container.y = options.yInit;
       
   871     options.parentContainer.addChild(this.container);
       
   872 
       
   873     var orientation = options.orientation;
       
   874     var isHorizontal = (orientation !== 'vertical');
       
   875 
       
   876     this.linesDown = options.linesDown;
       
   877     this.height = options.height;
       
   878     this.pixelsPerSecond = options.pixelsPerSecond;
       
   879     this.width = options.width;
       
   880     this.noteColors = options.noteColors;
       
   881     this.colorsReg = options.colorsReg || {};
       
   882     this.lineColor = options.lineColor;
       
   883     this.lineInterval = options.lineInterval;
       
   884     this.offsetMusic = options.offsetMusic || false;
       
   885     this.noteHeight = options.noteHeight;
       
   886     this.noteDict = {};
       
   887     this.startTs = options.startTs || Date.now();
       
   888 
       
   889     var started = false;
       
   890 
       
   891     var isHidden = function(child) {
       
   892         // TODO: the origin point is an approximation. Should refine this
       
   893         var globalPos = child.toGlobal(new PIXI.Point(0,0));
       
   894         return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ;
       
   895     };
       
   896 
       
   897     //TODO: I do not like the "regColor" object. This should not be global, but local
       
   898     this.getColor = function(canal) {
       
   899         var color = this.colorsReg[canal];
       
   900         if(typeof(color) === 'undefined') {
       
   901             var colorsRegSize = Object.keys(this.colorsReg).length;
       
   902             if(colorsRegSize < this.noteColors.length) {
       
   903                 color = this.colorsReg[canal] = this.noteColors[colorsRegSize];
       
   904             }
       
   905             else {
       
   906                 color = this.colorsReg[canal] = parseInt(randomColor({ luminosity: 'light', hue: 'random', format:'hex'}).replace(/^#/, ''), 16);
       
   907             }
       
   908         }
       
   909         return color;
       
   910     };
       
   911 
       
   912     this.getNoteRect = function(x, y, color, alpha, width, height) {
       
   913         var graphics = new PIXI.Graphics();
       
   914         graphics.beginFill(color, alpha);
       
   915         graphics.drawRect(0, 0, width, height);
       
   916         graphics.endFill();
       
   917         graphics.x = x;
       
   918         graphics.y = y;
       
   919         graphics.width = width;
       
   920         graphics.height = height;
       
   921         return graphics;
       
   922     };
       
   923 
       
   924     this.addNoteRaw = function(data) {
       
   925     	console.log(data);
       
   926         var note = data.content[3];
       
   927         var velocity = data.content[4];
       
   928         var ts = (data.content[0] - NTP_EPOCH_DELTA)*1000;
       
   929         var channel = data.content[2];
       
   930         var sessionTs = data.content[1];
       
   931 
       
   932         this.addNote(note, ts, sessionTs, velocity, channel, 0);
       
   933     };
       
   934 
       
   935     this.addNote = function(note, startTime, sessionTs, velocity, channel, duration) {
       
   936 
       
   937         var ts = startTime;
       
   938         if(this.offsetMusic) {
       
   939             ts = this.startTs + sessionTs;
       
   940         }
       
   941 
       
   942         var noteDuration = duration;
       
   943         var noteVelocity = velocity;
       
   944         var graphics;
       
   945         if(!duration) {
       
   946             if(typeof this.noteDict[channel]==='undefined'){
       
   947                 this.noteDict[channel] = {};
       
   948             }
       
   949             if(velocity===0) {
       
   950                 if(typeof this.noteDict[channel][note] !== 'undefined') {
       
   951                     var noteDef = this.noteDict[channel][note];
       
   952                     delete this.noteDict[channel][note];
       
   953                     noteDuration = sessionTs - noteDef.sessionTs;
       
   954                     graphics = noteDef.graphics;
       
   955                     noteVelocity = noteDef.velocity;
       
   956                     ts = noteDef.ts;
       
   957                 }
       
   958             }
       
   959             else {
       
   960                 noteDuration = Date.now() - ts;
       
   961                 this.noteDict[channel][note] = { ts: ts, velocity: velocity, sessionTs: sessionTs};
       
   962             }
       
   963         }
       
   964 
       
   965 
       
   966         if(!this.offsetMusic || velocity===0) {
       
   967 
       
   968             var width = noteDuration * this.pixelsPerSecond / 1000;
       
   969             if(!graphics) {
       
   970                 var x = (ts-this.startTs) * this.pixelsPerSecond / 1000;
       
   971                 if((x+width) <  (Math.abs(this.container.x) - this.width)) {
       
   972                     // not visible. do nothing
       
   973                     return;
       
   974                 }
       
   975                 var y = Math.floor((128-note+0.5) * this.height / 128 - (this.noteHeight/2));
       
   976                 var color = this.getColor(channel);
       
   977                 var alpha = (noteVelocity / 128);
       
   978 
       
   979                 graphics = this.getNoteRect(x, y, color, alpha, width, this.noteHeight);
       
   980                 this.container.addChild(graphics);
       
   981             }
       
   982             else {
       
   983                 graphics.width = width;
       
   984             }
       
   985 
       
   986             if(!duration && velocity) {
       
   987                 this.noteDict[channel][note].graphics = graphics;
       
   988             }
       
   989         }
       
   990     };
       
   991 
       
   992     this.addLine = function(ts){
       
   993 
       
   994         if(typeof(ts) === 'undefined') {
       
   995             ts = new Date();
       
   996         }
       
   997         var x = -this.container.x;
       
   998         var y = this.linesDown ? this.height - 20 : 0;
       
   999 
       
  1000         var graphics = new PIXI.Graphics()
       
  1001             .beginFill(0xFFFF00)
       
  1002             .lineStyle(1, this.lineColor)
       
  1003             .moveTo(0, 0)
       
  1004             .lineTo(0, 20)
       
  1005             .endFill();
       
  1006         graphics.x = x;
       
  1007         graphics.y = y;
       
  1008         this.container.addChild(graphics);
       
  1009         // Add text
       
  1010         //var totalSec = lineNb * this.lineInterval / 1000;
       
  1011         var hours = ts.getHours();
       
  1012         var minutes =ts.getMinutes();
       
  1013         var seconds = ts.getSeconds();
       
  1014         var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds  < 10 ? '0' + seconds : seconds);
       
  1015 
       
  1016         var fontObj = { font: '10pt Arial', fill: '#444444' };
       
  1017         var t = new PIXI.Text(timeStr, fontObj);
       
  1018         if(isHorizontal) {
       
  1019             t.x = x + 2;
       
  1020             t.y = this.linesDown ? this.height - 15 : 2;
       
  1021         }
       
  1022         else {
       
  1023             t.rotation = -Math.PI/2;
       
  1024             t.x = x ;
       
  1025             t.y = this.linesDown ? this.height - 2 : t.width + 2;
       
  1026         }
       
  1027         this.container.addChild(t);
       
  1028     };
       
  1029 
       
  1030     this.moveTo = function(diffTime){
       
  1031         var oldX = this.container.x;
       
  1032         this.container.x = Math.floor(diffTime*this.pixelsPerSecond);
       
  1033         var deltaX = Math.abs(oldX-this.container.x);
       
  1034         _.forOwn(this.noteDict, function(channelDict) {
       
  1035             _.forOwn(channelDict, function(noteDef) {
       
  1036                 if(noteDef.graphics) {
       
  1037                     noteDef.graphics.width = noteDef.graphics.width + deltaX;
       
  1038                 }
       
  1039             });
       
  1040         });
       
  1041     };
       
  1042 
       
  1043     this.move = function() {
       
  1044         var diff = (this.startTs - Date.now())/1000;
       
  1045         this.moveTo(diff);
       
  1046     };
       
  1047 
       
  1048     this.removePassedObjets = function(){
       
  1049         var childrenToRemove = [];
       
  1050         _(_this.container.children).forEach(function(child) {
       
  1051             return typeof(child) === 'undefined' ||
       
  1052                 (isHidden(child) && childrenToRemove.push(child));
       
  1053         });
       
  1054         childrenToRemove.forEach(function(child) {
       
  1055             _this.container.removeChild(child);
       
  1056         });
       
  1057     };
       
  1058 
       
  1059     this.start = function() {
       
  1060         if(!started) {
       
  1061             this.startTs = Date.now();
       
  1062             this.addLine();
       
  1063             started = true;
       
  1064         }
       
  1065         this.verticalLinesInterval = setInterval(function() { _this.addLine(); }, this.lineInterval);
       
  1066         this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.width / this.pixelsPerSecond );
       
  1067     };
       
  1068 
       
  1069     this.stop = function() {
       
  1070         //window.clearInterval(this.moveInterval);
       
  1071         clearInterval(this.verticalLinesInterval);
       
  1072         clearInterval(this.cleanInterval);
       
  1073     };
       
  1074 
       
  1075 
       
  1076 }
       
  1077 
       
  1078 module.exports = PianoRoll;
       
  1079 
       
  1080 },{"lodash":"lodash","pixi":"pixi","randomColor":"randomColor"}],8:[function(require,module,exports){
       
  1081 /**
       
  1082 * scripts/stageview.js
       
  1083 *
       
  1084 * This is the starting point for your application.
       
  1085 * Take a look at http://browserify.org/ for more info
       
  1086 */
       
  1087 
       
  1088 /* global document: false */
       
  1089 
       
  1090 'use strict';
       
  1091 
       
  1092 
       
  1093 var PIXI = require('pixi');
       
  1094 var _ = require('lodash');
       
  1095 
       
  1096 var defaultConfig = {
       
  1097     externalRefresh: false,
       
  1098     logger: undefined,
       
  1099     sceneWidth: 1024,
       
  1100     sceneHeight: 768,
       
  1101     framerate: 25,
       
  1102     sceneBgColor: 0xFFFFFF,
       
  1103     canvasContainer: 'canvasContainer',
       
  1104 };
       
  1105 
       
  1106 function StageView(options) {
       
  1107 
       
  1108     var _this = this;
       
  1109     var opts = _(options).defaults(defaultConfig).value();
       
  1110 
       
  1111     var externalRefresh = opts.externalRefresh;
       
  1112 
       
  1113     this.logger = opts.logger;
       
  1114     this.framerate = opts.framerate;
       
  1115     var sceneBgColor = opts.sceneBgColor;
       
  1116     var sceneWidth = opts.sceneWidth;
       
  1117     var sceneHeight = opts.sceneHeight;
       
  1118     var canvasContainer = opts.canvasContainer;
       
  1119     var timeContainer = [];
       
  1120     var components = []; 
       
  1121     
       
  1122     //create an new instance of a pixi stage
       
  1123     this.stage = new PIXI.Stage(sceneBgColor);
       
  1124     //create a renderer instance.
       
  1125     var renderer = PIXI.autoDetectRenderer(sceneWidth, sceneHeight);
       
  1126     	
       
  1127     this.init = function() {
       
  1128 
       
  1129         if(typeof(canvasContainer) === 'string') {
       
  1130             canvasContainer = document.getElementById(canvasContainer);
       
  1131         }
       
  1132         if(typeof(timeContainer) === 'string') {
       
  1133             timeContainer = document.getElementById(timeContainer);
       
  1134         }
       
  1135 
       
  1136         canvasContainer.appendChild(renderer.view);
       
  1137         
       
  1138         components.forEach(function(c){
       
  1139     		c.init();
       
  1140     	});
       
  1141     };
       
  1142     
       
  1143     this.registerTimeContainer = function(container) {
       
  1144     	timeContainer.push(container);
       
  1145     };
       
  1146     
       
  1147     this.registerComponent = function(component) {
       
  1148     	components.push(component);
       
  1149     	this.stage.addChild(component.container);
       
  1150     };
       
  1151 
       
  1152     this.refresh = function() {
       
  1153     	components.forEach(function(c){
       
  1154     		c.refresh();
       
  1155     	});
       
  1156         renderer.render(this.stage);
       
  1157     };
       
  1158 
       
  1159     // Init page and intervals
       
  1160     var refreshInterval;
       
  1161 
       
  1162     this.start = function() {
       
  1163 
       
  1164         if(!externalRefresh) {
       
  1165             refreshInterval = setInterval(function() {_this.refresh();}, 1000/this.framerate);
       
  1166         }
       
  1167         
       
  1168         components.forEach(function(c){
       
  1169     		c.start();
       
  1170     	});
       
  1171     };
       
  1172 
       
  1173     this.stop = function() {
       
  1174         if(!externalRefresh) {
       
  1175             clearInterval(refreshInterval);
       
  1176         }
       
  1177         clearInterval(refreshTimeInterval);
       
  1178         
       
  1179         components.forEach(function(c){
       
  1180     		c.stop();
       
  1181     	});
       
  1182     };
       
  1183 
       
  1184 
       
  1185     this.log = function(m) {
       
  1186         if(this.logger) {
       
  1187             this.logger.log(m);
       
  1188         }
       
  1189     };
       
  1190 
       
  1191 
       
  1192     return this;
       
  1193 }
       
  1194 
       
  1195 module.exports = {
       
  1196     StageView: StageView
       
  1197 };
       
  1198 
       
  1199 },{"lodash":"lodash","pixi":"pixi"}],9:[function(require,module,exports){
       
  1200 /**
       
  1201 * js/utils.js
       
  1202 *
       
  1203 * basic tools
       
  1204 *
       
  1205 */
       
  1206 
       
  1207 'use strict';
       
  1208 
       
  1209 function formatTime (ts) {
       
  1210 	var hours = Math.floor( (ts/1000) / 3600 ) % 24;
       
  1211 	var minutes = Math.floor( (ts/1000) / 60 ) % 60;
       
  1212 	var seconds = Math.floor( (ts/1000) % 60);
       
  1213 	return ((hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds  < 10 ? '0' + seconds : seconds));
       
  1214 }
       
  1215 
       
  1216 
       
  1217 module.exports = {
       
  1218 	formatTime: formatTime
       
  1219 };
       
  1220 
       
  1221 },{}],10:[function(require,module,exports){
       
  1222 /**
       
  1223 * js/wswrapper.js
       
  1224 *
       
  1225 * simple webservice wrapper to register callbacks on onmessage
       
  1226 *
       
  1227 */
       
  1228 
       
  1229 /* global WebSocket: false */
       
  1230 
       
  1231 'use strict';
       
  1232 
       
  1233 function WsWrapper(wsurl, logger) {
       
  1234 
       
  1235     var url = wsurl;
       
  1236     var sock = new WebSocket(url);
       
  1237     var loggerObj = logger;
       
  1238 
       
  1239     var log = function(msg) {
       
  1240         if(loggerObj) {
       
  1241             loggerObj.log(msg);
       
  1242         }
       
  1243     };
       
  1244 
       
  1245     var handlers = [];
       
  1246 
       
  1247     sock.onopen = function() {
       
  1248         log('Connected to ' + url);
       
  1249     };
       
  1250 
       
  1251     sock.onclose = function(e) {
       
  1252         log('Connection closed (wasClean = ' + e.wasClean + ', code = ' + e.code + ', reason = \'' + e.reason + '\')');
       
  1253         sock = null;
       
  1254     };
       
  1255 
       
  1256     sock.onmessage = function(e) {
       
  1257         log('received ' + e.data);
       
  1258         var data = JSON.parse(e.data);
       
  1259         handlers.forEach(function(handler) {
       
  1260             handler(data);
       
  1261         });
       
  1262     };
       
  1263 
       
  1264     this.message = function(handler) {
       
  1265         if(handler) {
       
  1266             handlers.push(handler);
       
  1267         }
       
  1268     };
       
  1269 
       
  1270 }
       
  1271 
       
  1272 module.exports = {
       
  1273     WsWrapper: WsWrapper
       
  1274 };
       
  1275 
       
  1276 },{}]},{},[1])(1)
       
  1277 });
       
  1278 //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5vZGVfbW9kdWxlcy9icm93c2VyaWZ5L25vZGVfbW9kdWxlcy9icm93c2VyLXBhY2svX3ByZWx1ZGUuanMiLCIuL2FwcC9qcy9tYWluLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9hbm5vdHNyb2xsLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9hbm5vdHN0aW1lbGluZS5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvYW5ub3Rzdml6dmlldy5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvZG91Ymxlcm9sbC5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvbG9nZ2VyLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9waWFub3JvbGwuanMiLCIvVXNlcnMveW1oL2Rldi9wcm9qZWN0cy9tb25zL2Rldi9jbGllbnQvYW5ub3R2aXovYXBwL2pzL3N0YWdldmlldy5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvdXRpbHMuanMiLCIvVXNlcnMveW1oL2Rldi9wcm9qZWN0cy9tb25zL2Rldi9jbGllbnQvYW5ub3R2aXovYXBwL2pzL3dzd3JhcHBlci5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtBQ0FBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzNCQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDOU1BO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3JOQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUM1SkE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDN0xBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzdDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3JPQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUNySEE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3BCQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJmaWxlIjoiZ2VuZXJhdGVkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXNDb250ZW50IjpbIihmdW5jdGlvbiBlKHQsbixyKXtmdW5jdGlvbiBzKG8sdSl7aWYoIW5bb10pe2lmKCF0W29dKXt2YXIgYT10eXBlb2YgcmVxdWlyZT09XCJmdW5jdGlvblwiJiZyZXF1aXJlO2lmKCF1JiZhKXJldHVybiBhKG8sITApO2lmKGkpcmV0dXJuIGkobywhMCk7dmFyIGY9bmV3IEVycm9yKFwiQ2Fubm90IGZpbmQgbW9kdWxlICdcIitvK1wiJ1wiKTt0aHJvdyBmLmNvZGU9XCJNT0RVTEVfTk9UX0ZPVU5EXCIsZn12YXIgbD1uW29dPXtleHBvcnRzOnt9fTt0W29dWzBdLmNhbGwobC5leHBvcnRzLGZ1bmN0aW9uKGUpe3ZhciBuPXRbb11bMV1bZV07cmV0dXJuIHMobj9uOmUpfSxsLGwuZXhwb3J0cyxlLHQsbixyKX1yZXR1cm4gbltvXS5leHBvcnRzfXZhciBpPXR5cGVvZiByZXF1aXJlPT1cImZ1bmN0aW9uXCImJnJlcXVpcmU7Zm9yKHZhciBvPTA7bzxyLmxlbmd0aDtvKyspcyhyW29dKTtyZXR1cm4gc30pIiwiLyoqXG4gKiBzY3JpcHRzL21haW4uanNcbiAqXG4gKiBUaGlzIGlzIHRoZSBzdGFydGluZyBwb2ludCBmb3IgeW91ciBhcHBsaWNhdGlvbi5cbiAqIFRha2UgYSBsb29rIGF0IGh0dHA6Ly9icm93c2VyaWZ5Lm9yZy8gZm9yIG1vcmUgaW5mb1xuICovXG5cbid1c2Ugc3RyaWN0JztcblxudmFyIGRvdWJsZXJvbGwgPSByZXF1aXJlKCcuL2RvdWJsZXJvbGwnKTtcbnZhciBhbm5vdHNyb2xsID0gcmVxdWlyZSgnLi9hbm5vdHNyb2xsJyk7XG52YXIgYW5ub3RzdGltZWxpbmUgPSByZXF1aXJlKCcuL2Fubm90c3RpbWVsaW5lJyk7XG52YXIgYW5ub3Rzdml6dmlldyA9IHJlcXVpcmUoJy4vYW5ub3Rzdml6dmlldycpO1xudmFyIHN0YWdldmlldyA9IHJlcXVpcmUoJy4vc3RhZ2V2aWV3Jyk7XG52YXIgd3N3cmFwcGVyID0gcmVxdWlyZSgnLi93c3dyYXBwZXInKTtcbnZhciBsb2dnZXIgPSByZXF1aXJlKCcuL2xvZ2dlcicpO1xuXG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xuXG5tb2R1bGUuZXhwb3J0cyA9IF8oe30pXG4gICAgLmV4dGVuZChkb3VibGVyb2xsKVxuICAgIC5leHRlbmQoYW5ub3Rzcm9sbClcbiAgICAuZXh0ZW5kKGFubm90c3RpbWVsaW5lKVxuICAgIC5leHRlbmQoYW5ub3Rzdml6dmlldylcbiAgICAuZXh0ZW5kKHN0YWdldmlldylcbiAgICAuZXh0ZW5kKHdzd3JhcHBlcilcbiAgICAuZXh0ZW5kKGxvZ2dlcilcbiAgICAudmFsdWUoKTsiLCIvKipcbioganMvYW5ub3RzUm9sbC5qc1xuKlxuKiBhbm5vdHNSb2xsIGJhc2ljIGNvbXBvbmVudFxuKlxuKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG5cbnZhciBERUZBVUxUX0FOTk9UX0NPTE9SID0gJyNiYWJhYmEnO1xuXG52YXIgZGVmYXVsdEFubm90U3R5bGVzID0ge1xuICAgICdsYWJlbCc6IHsgZm9udDogJzE2cHQgQXJpYWwgQm9sZCcsIGZpbGw6ICcjNjVBOTU0Jywgd29yZFdyYXA6IHRydWV9LFxuICAgICd0ZXh0JyA6IHsgZm9udDogJzEycHQgQXJpYWwgUmVndWxhcicsIGZpbGw6ICcjNDQ0NDQ0Jywgd29yZFdyYXA6IHRydWV9LFxuICAgICd1c2VyJyA6IHsgZm9udDogJzE0cHQgQXJpYWwgcmVndWxhcicsIGZpbGw6ICcjNjY2NjY2JyB9LFxufTtcblxudmFyIGRlZmF1bHRPcHRpb25zID0ge1xuICAgIGV4dGVybmFsUmVmcmVzaDogZmFsc2UsXG4gICAgZGVmYXVsdENvbG9yOiBERUZBVUxUX0FOTk9UX0NPTE9SLFxuICAgIGFubm90U3R5bGVzOiBkZWZhdWx0QW5ub3RTdHlsZXNcbn07XG5cbmZ1bmN0aW9uIEFubm90c1JvbGwob3B0aW9ucykge1xuXG4vL3BhcmVudENvbnRhaW5lciwgeEluaXQsIHlJbml0LCB3aWR0aCwgaGVpZ2h0LCB3aWR0aFJvbGwsIHBpeGVsc1BlclNlY29uZCwgYW5ub3RDb2xvcnNcbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0T3B0aW9ucykudmFsdWUoKTtcblxuXG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IG9wdHMueEluaXQ7XG4gICAgdGhpcy5jb250YWluZXIueSA9IG9wdHMueUluaXQ7XG4gICAgdGhpcy5jb250YWluZXIud2lkdGggPSBvcHRzLndpZHRoO1xuXG4gICAgdGhpcy5oZWlnaHQgPSBvcHRzLmhlaWdodDtcbiAgICB0aGlzLndpZHRoID0gb3B0cy53aWR0aDtcbiAgICB0aGlzLndpZHRoUm9sbCA9IG9wdHMud2lkdGhSb2xsO1xuICAgIHRoaXMucGl4ZWxzUGVyU2Vjb25kID0gb3B0cy5waXhlbHNQZXJTZWNvbmQ7XG4gICAgdGhpcy5hbm5vdENvbG9ycyA9IG9wdHMuYW5ub3RDb2xvcnM7XG4gICAgdGhpcy5zdGFydFRzID0gb3B0aW9ucy5zdGFydFRzIHx8IERhdGUubm93KCk7XG5cbiAgICB2YXIgeUluaXQgPSBvcHRzLnlJbml0O1xuICAgIHZhciBhbm5vdFN0eWxlcyA9IF8ob3B0cy5hbm5vdFN0eWxlcykuZGVmYXVsdHMoZGVmYXVsdEFubm90U3R5bGVzKS52YWx1ZSgpO1xuICAgIGZvcih2YXIgc3R5bGUgaW4gYW5ub3RTdHlsZXMpIHtcbiAgICBcdGlmIChhbm5vdFN0eWxlc1tzdHlsZV0ud29yZFdyYXAgPT09IHRydWUpe1xuICAgIFx0XHRhbm5vdFN0eWxlc1tzdHlsZV0ud29yZFdyYXBXaWR0aCA9IHRoaXMud2lkdGhSb2xsOyBcbiAgICBcdH1cbiAgICB9XG4gICAgY29uc29sZS5sb2coYW5ub3RTdHlsZXMpO1xuICAgIHZhciBzdGFydGVkID0gZmFsc2U7XG4gICAgdmFyIHdzID0gb3B0cy53cztcbiAgICB2YXIgZXh0ZXJuYWxSZWZyZXNoID0gb3B0cy5leHRlcm5hbFJlZnJlc2g7XG4gICAgdmFyIHN0YWdlVmlldyA9IG9wdHMuc3RhZ2VWaWV3O1xuICAgIFxuICAgIHN0YWdlVmlldy5yZWdpc3RlckNvbXBvbmVudCh0aGlzKTtcblxuICAgIHZhciBpc0hpZGRlbiA9IGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgICAgIC8vIFRPRE86IHRoZSBvcmlnaW4gcG9pbnQgaXMgYW4gYXBwcm94aW1hdGlvbi4gU2hvdWxkIHJlZmluZSB0aGlzXG4gICAgICAgIHZhciBnbG9iYWxQb3MgPSBjaGlsZC50b0dsb2JhbChuZXcgUElYSS5Qb2ludCgwLDApKTtcbiAgICAgICAgcmV0dXJuICgoZ2xvYmFsUG9zLnggKyBjaGlsZC53aWR0aCkgPCAwKSB8fCAoKGdsb2JhbFBvcy55ICsgY2hpbGQuaGVpZ2h0KSA8IDApIDtcbiAgICB9O1xuXG4gICAgdGhpcy5hZGRBbm5vdHMgPSBmdW5jdGlvbihkYXRhKSB7XG5cbiAgICAgICAgLy92YXIgdGl0bGUgPSBkYXRhLmNvbnRlbnQuY2F0ZWdvcnkubGFiZWw7XG4gICAgICAgIC8vdmFyIHVzZXIgPSBkYXRhLmNvbnRlbnQudXNlcjtcbiAgICAgICAgLy9UZXN0IGNhdCBhbmQgY29sb3JcbiAgICAgICAgLy92YXIgY29sb3JBbm5vdCA9IDB4NjVBOTU0O1xuICAgICAgICB2YXIgY2F0ZWdvcnkgPSBkYXRhLmNvbnRlbnQuY2F0ZWdvcnkubGFiZWwsXG4gICAgICAgICAgICB0ZXh0ICAgICA9IGRhdGEuY29udGVudC50ZXh0LFxuICAgICAgICAgICAgdXNlciAgICAgPSBkYXRhLmNvbnRlbnQudXNlcixcbiAgICAgICAgICAgIHRzICAgICAgID0gRGF0ZS5wYXJzZShkYXRhLnRzKSxcbiAgICAgICAgICAgIGNvbG9yICAgID0gdGhpcy5nZXRDb2xvcih0cywgZGF0YS5jb250ZW50LmNhdGVnb3J5LmNvZGUpO1xuXG4gICAgICAgIHRoaXMuYWRkQW5ub3QoY2F0ZWdvcnksIHRleHQsIHVzZXIsIGNvbG9yLCB0cyk7XG4gICAgfTtcblxuICAgIHRoaXMuZ2V0Q29sb3IgPSBmdW5jdGlvbih0cywgY29kZSkge1xuICAgICAgICB2YXIgY29sb3JzRGVmO1xuICAgICAgICBfKHRoaXMuYW5ub3RDb2xvcnMpLmVhY2hSaWdodChmdW5jdGlvbihjZGVmKSB7XG4gICAgICAgICAgICBjb25zb2xlLmxvZyhcImNEZWZcIiwgY2RlZik7XG4gICAgICAgICAgICBjb25zb2xlLmxvZyhcImNEZWYgdHNcIiwgY2RlZi50cywgdHMpO1xuICAgICAgICAgICAgaWYoY2RlZi50cyA8IHRzKSB7XG4gICAgICAgICAgICAgICAgY29sb3JzRGVmID0gY2RlZi5jb2xvcnM7XG4gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICAgICAgfVxuICAgICAgICB9KTtcbiAgICAgICAgdmFyIHJlc0NvbG9yO1xuICAgICAgICBjb25zb2xlLmxvZyhcImNvbG9yc0RlZlwiLCBjb2xvcnNEZWYpO1xuICAgICAgICBpZihjb2xvcnNEZWYpIHtcbiAgICAgICAgICAgIHJlc0NvbG9yID0gY29sb3JzRGVmW2NvZGVdO1xuICAgICAgICB9XG4gICAgICAgIGlmKCFyZXNDb2xvcikge1xuICAgICAgICAgICAgcmVzQ29sb3IgPSBERUZBVUxUX0FOTk9UX0NPTE9SO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiByZXNDb2xvcjtcbiAgICB9XG5cbiAgICB0aGlzLmFkZEFubm90ID0gZnVuY3Rpb24oY2F0ZWdvcnksIHRleHQsIHVzZXIsIGNhdENvbG9yLCB0cyl7XG5cbiAgICAgICAgdmFyIGNvbG9yID0gY2F0Q29sb3IgPyBjYXRDb2xvciA6IERFRkFVTFRfQU5OT1RfQ09MT1I7XG4gICAgICAgIHZhciB4ID0gMDtcbiAgICAgICAgdmFyIHkgPSAodHMtdGhpcy5zdGFydFRzKSAqIHRoaXMucGl4ZWxzUGVyU2Vjb25kIC8gMTAwMCArIHlJbml0O1xuXG4gICAgICAgIHZhciBjb2xvckhleCA9IHBhcnNlSW50KGNvbG9yLnJlcGxhY2UoL14jLywgJycpLCAxNik7XG5cbiAgICAgICAgdmFyIGdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKVxuICAgICAgICAgICAgLmJlZ2luRmlsbChjb2xvckhleClcbiAgICAgICAgICAgIC5kcmF3UmVjdCh4LCB5LCAxMCwgMylcbiAgICAgICAgICAgIC5lbmRGaWxsKCk7XG5cbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoZ3JhcGhpY3MpO1xuXG4gICAgICAgIHZhciB0ZXh0SGVpZ2h0ID0gMDtcbiAgICAgICAgdmFyIGNhdExhYmVsID0gbmV3IFBJWEkuVGV4dChcbiAgICAgICAgICAgIGNhdGVnb3J5LFxuICAgICAgICAgICAgXyhhbm5vdFN0eWxlcy5sYWJlbCkuZXh0ZW5kKHtmaWxsOiBjb2xvcn0pLnZhbHVlKClcbiAgICAgICAgKTtcbiAgICAgICAgY2F0TGFiZWwueCA9IHggKyAyMDtcbiAgICAgICAgY2F0TGFiZWwueSA9IHkgLSAyMztcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY2F0TGFiZWwpO1xuICAgICAgICB0ZXh0SGVpZ2h0ICs9IChjYXRMYWJlbC5oZWlnaHQgLSAyMyArIDIpO1xuXG4gICAgICAgIGlmKHRleHQpIHtcbiAgICAgICAgICAgIHZhciBjYXRUZXh0ID0gbmV3IFBJWEkuVGV4dCh0ZXh0LCBhbm5vdFN0eWxlcy50ZXh0KTtcbiAgICAgICAgICAgIGNhdFRleHQueCA9IHggKyAyMDtcbiAgICAgICAgICAgIGNhdFRleHQueSA9IHkgKyAyO1xuICAgICAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY2F0VGV4dCk7XG4gICAgICAgICAgICB0ZXh0SGVpZ2h0ICs9IChjYXRUZXh0LmhlaWdodCArIDIpO1xuICAgICAgICB9XG5cbiAgICAgICAgdmFyIGNhdFVzZXIgPSBuZXcgUElYSS5UZXh0KHVzZXIsIGFubm90U3R5bGVzLnVzZXIpO1xuICAgICAgICBjYXRVc2VyLnggPSB4ICsgMjA7XG4gICAgICAgIGNhdFVzZXIueSA9IHkgKyAyICsgdGV4dEhlaWdodDtcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY2F0VXNlcik7XG5cbiAgICAgICAgdGhpcy5hZGRBbm5vdExpbmUoY29sb3JIZXgsIHkpO1xuICAgIH07XG5cbiAgICB0aGlzLmFkZEFubm90TGluZSA9IGZ1bmN0aW9uKGNvbG9yLCB5KSB7XG4gICAgICAgIHZhciB4ID0gdGhpcy53aWR0aFJvbGw7XG5cblxuICAgICAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpXG4gICAgICAgICAgICAuYmVnaW5GaWxsKGNvbG9yKVxuICAgICAgICAgICAgLmRyYXdSZWN0KHgsIHksIHRoaXMud2lkdGggLSB4LCAzKVxuICAgICAgICAgICAgLmVuZEZpbGwoKTtcblxuICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChncmFwaGljcyk7XG4gICAgfTtcblxuICAgIHRoaXMubW92ZVRvID0gZnVuY3Rpb24oZGlmZlRpbWUpe1xuICAgIFx0dGhpcy5jb250YWluZXIueSA9IE1hdGguZmxvb3IoZGlmZlRpbWUqdGhpcy5waXhlbHNQZXJTZWNvbmQpO1xuICAgIH07XG5cbiAgICB0aGlzLm1vdmUgPSB0aGlzLnJlZnJlc2ggPSBmdW5jdGlvbigpIHtcbiAgICAgICAgdmFyIGRpZmYgPSAodGhpcy5zdGFydFRzIC0gRGF0ZS5ub3coKSkvMTAwMDtcbiAgICAgICAgdGhpcy5tb3ZlVG8oZGlmZik7XG4gICAgfTtcblxuICAgIHRoaXMucmVtb3ZlUGFzc2VkT2JqZXRzID0gZnVuY3Rpb24oKXtcbiAgICAgICAgdmFyIGNoaWxkcmVuVG9SZW1vdmUgPSBbXTtcbiAgICAgICAgXyhfdGhpcy5jb250YWluZXIuY2hpbGRyZW4pLmZvckVhY2goZnVuY3Rpb24oY2hpbGQpIHtcbiAgICAgICAgICAgIHJldHVybiB0eXBlb2YoY2hpbGQpID09PSAndW5kZWZpbmVkJyB8fFxuICAgICAgICAgICAgICAgIChpc0hpZGRlbihjaGlsZCkgJiYgY2hpbGRyZW5Ub1JlbW92ZS5wdXNoKGNoaWxkKSk7XG4gICAgICAgIH0pO1xuICAgICAgICBjaGlsZHJlblRvUmVtb3ZlLmZvckVhY2goZnVuY3Rpb24oY2hpbGQpIHtcbiAgICAgICAgICAgIF90aGlzLmNvbnRhaW5lci5yZW1vdmVDaGlsZChjaGlsZCk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cbiAgICB0aGlzLmluaXQgPSBmdW5jdGlvbigpIHtcblxuICAgICAgICB3cy5tZXNzYWdlKGZ1bmN0aW9uKGRhdGEpIHtcbiAgICAgICAgICAgIF90aGlzLmFkZEFubm90cyhkYXRhKTtcbiAgICAgICAgfSk7XG5cbiAgICB9O1xuXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBpZighc3RhcnRlZCkge1xuICAgICAgICAgICAgdGhpcy5zdGFydFRzID0gRGF0ZS5ub3coKTtcbiAgICAgICAgICAgIHN0YXJ0ZWQgPSB0cnVlO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMuY2xlYW5JbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uICgpIHsgX3RoaXMucmVtb3ZlUGFzc2VkT2JqZXRzKCk7IH0sIDEwMDAgKiB0aGlzLmhlaWdodCAvIHRoaXMucGl4ZWxzUGVyU2Vjb25kICk7XG4gICAgICAgIGlmKCFleHRlcm5hbFJlZnJlc2gpIHtcbiAgICAgICAgICAgIHRoaXMucmVmcmVzaEludGVydmFsID0gc2V0SW50ZXJ2YWwoZnVuY3Rpb24oKSB7X3RoaXMubW92ZSgpO30sIDEwMDAvdGhpcy5mcmFtZXJhdGUpO1xuICAgICAgICB9XG4gICAgfTtcblxuICAgIHRoaXMuc3RvcCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBjbGVhckludGVydmFsKHRoaXMuY2xlYW5JbnRlcnZhbCk7XG4gICAgICAgIGlmKCFleHRlcm5hbFJlZnJlc2gpIHtcbiAgICAgICAgICAgIGNsZWFySW50ZXJ2YWwodGhpcy5yZWZyZXNoSW50ZXJ2YWwpO1xuICAgICAgICB9XG4gICAgfTtcblxufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBBbm5vdHNSb2xsOiBBbm5vdHNSb2xsLFxufTtcbiIsIi8qKlxuKiBqcy9hbm5vdHN0aW1lbGluZVxuKlxuKiBhbm5vdHN0aW1lbGluZSBiYXNpYyBjb21wb25lbnRcbipcbiovXG5cbid1c2Ugc3RyaWN0JztcblxudmFyIFBJWEkgPSByZXF1aXJlKCdwaXhpJyk7XG52YXIgVXRpbHMgPSByZXF1aXJlKCcuL3V0aWxzLmpzJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xuXG52YXIgZGVmYXVsdE9wdGlvbnMgPSB7XHRcdFxuICAgIGxvZ2dlcjogdW5kZWZpbmVkLFxuICAgIGludGVydmFsV2lkdGg6IDEwLFxuICAgIGludGVydmFsSGVpZ2h0OiA1LFxuICAgIG1heENlbGxIZWlnaHQ6IDIwMCxcbiAgICByYWRpdXM6IDMwMFxufTtcblxuXG5mdW5jdGlvbiBBbm5vdHNUaW1lTGluZShvcHRpb25zKXtcbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0T3B0aW9ucykudmFsdWUoKTtcbiAgICBcbiAgICB0aGlzLmNvbnRhaW5lciA9IG5ldyBQSVhJLkRpc3BsYXlPYmplY3RDb250YWluZXIoKTtcbiAgICB0aGlzLmNvbnRhaW5lci54ID0gb3B0cy54SW5pdDtcbiAgICB0aGlzLmNvbnRhaW5lci55ID0gb3B0cy55SW5pdDtcbiAgICB0aGlzLmNvbnRhaW5lci53aWR0aCA9IG9wdHMud2lkdGg7XG4gICAgdGhpcy5jb250YWluZXIuaGVpZ2h0ID0gb3B0cy5oZWlnaHQ7ICAgIFxuICAgIFxuICAgIHRoaXMudGltZUJlZ2luID0gb3B0cy50aW1lQmVnaW47XG4gICAgdGhpcy50aW1lRW5kID0gb3B0cy50aW1lRW5kO1xuICAgIHRoaXMuZHVyYXRpb24gPSAodGhpcy50aW1lRW5kIC0gdGhpcy50aW1lQmVnaW4pLzEwMDA7XG4gICAgdGhpcy53aWR0aCA9IG9wdHMud2lkdGg7XG4gICAgdGhpcy5oZWlnaHQgPSBvcHRzLmhlaWdodDtcbiAgICB0aGlzLmludGVydmFsSGVpZ2h0ID0gb3B0cy5pbnRlcnZhbEhlaWdodDtcbiAgICB0aGlzLmludGVydmFsV2lkdGggPSBvcHRzLmludGVydmFsV2lkdGg7XG4gICAgdGhpcy5tYXhDZWxsSGVpZ2h0ID0gb3B0cy5tYXhDZWxsSGVpZ2h0O1xuICAgIHRoaXMuYW5ub3RDYXRlZ29yaWVzID0gb3B0cy5hbm5vdENhdGVnb3JpZXM7XG4gICAgXG4gICAgdGhpcy5jaXJjbGVYID0gb3B0cy5jaXJjbGVYIHx8ICh0aGlzLndpZHRoLzIpO1xuICAgIHRoaXMuY2lyY2xlWSA9IG9wdHMuY2lyY2xlWSB8fCAodGhpcy5oZWlnaHQvMik7XG4gICAgdGhpcy5yYWRpdXMgPSBvcHRzLnJhZGl1cztcbiAgICB0aGlzLnBlcmltZXRlciA9IDIqTWF0aC5QSSogdGhpcy5yYWRpdXM7XG4gICAgdGhpcy5pbnRlcnZhbER1cmF0aW9uID0gKHRoaXMuaW50ZXJ2YWxXaWR0aCAqIHRoaXMuZHVyYXRpb24gLyB0aGlzLnBlcmltZXRlcik7XG4gICAgXG4gICAgdmFyIGN1cnJlbnRUaW1lID0gdGhpcy50aW1lQmVnaW47XG4gICAgdmFyIHRvdGFsSW5kZXggPSBNYXRoLmZsb29yKHRoaXMucGVyaW1ldGVyL3RoaXMuaW50ZXJ2YWxXaWR0aCk7XG4gICAgXHRcbiAgICB0aGlzLmNlbGxzID0gW11cbiAgICBmb3IgKHZhciBpPTA7IGk8KHRoaXMucGVyaW1ldGVyL3RoaXMuaW50ZXJ2YWxXaWR0aCkgOyBpKyspe1xuICAgIFx0dGhpcy5jZWxsc1tpXSA9IFtdO1xuICAgIFx0dGhpcy5jZWxsc1tpXS5pID0gaTtcbiAgICBcdHRoaXMuY2VsbHNbaV0udG90YWxBbm5vdHMgPSAwO1xuICAgIFx0dGhpcy5jZWxsc1tpXS5jYXRlZ29yaWVzID0ge307XG4gICAgXHRcbiAgICBcdGZvciAodmFyIGNhdGVnb3J5IGluIHRoaXMuYW5ub3RDYXRlZ29yaWVzWzBdLmNvbG9ycyl7XG4gICAgXHRcdHRoaXMuY2VsbHNbaV0uY2F0ZWdvcmllc1tjYXRlZ29yeV0gPSB7XG5cdFx0XHRcdFwiY291bnRcIjogMCxcblx0XHRcdFx0XCJjb2xvclwiOiB0aGlzLmFubm90Q2F0ZWdvcmllc1swXS5jb2xvcnNbY2F0ZWdvcnldXG4gICAgXHRcdH07XG4gICAgXHR9XG4gICAgfVxuICAgIFxuICAgIHZhciB3cyA9IG9wdHMud3M7XG4gICAgdmFyIHN0YWdlVmlldyA9IG9wdHMuc3RhZ2VWaWV3O1xuXG4gICAgLy9kcmF3IHRoZSBiYXNlIC0gY2lyY2xlIGFuZCBsaW5lIHRvIGxvY2F0ZSB0aGUgc2NlbmVcbiAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpO1xuICAgIGdyYXBoaWNzLmxpbmVTdHlsZSgyLCAweDY0NjQ2NClcbiAgICBcdC5kcmF3Q2lyY2xlKHRoaXMuY2lyY2xlWCwgdGhpcy5jaXJjbGVZLCB0aGlzLnJhZGl1cyAtIDMpXG4gICAgXHQubGluZVN0eWxlKDEsIDB4RDdEN0Q3KVxuICAgIFx0LmRyYXdDaXJjbGUodGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVksIHRoaXMucmFkaXVzKjIvMylcbiAgICBcdC5kcmF3Q2lyY2xlKHRoaXMuY2lyY2xlWCwgdGhpcy5jaXJjbGVZLCB0aGlzLnJhZGl1cy8zKVxuICAgIFx0LmxpbmVTdHlsZSgxLCAweDY0NjQ2NClcbiAgICBcdC5tb3ZlVG8odGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVkgLSAodGhpcy5yYWRpdXMvMykvMilcbiAgICBcdC5saW5lVG8odGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVkgLSB0aGlzLnJhZGl1cyAtIHRoaXMubWF4Q2VsbEhlaWdodCAtIDEwKVxuICAgIFx0LmVuZEZpbGwoKVxuICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGdyYXBoaWNzKTtcbiAgICBcbiAgICAvL3NldCB0aW1lIHRleHRcbiAgICB2YXIgY3VycmVudFRpbWVUZXh0ID0gbmV3IFBJWEkuVGV4dChcIi0tIDogLS0gOiAtLVwiLCB7IGZvbnQ6ICcxOHB0IEdvdGhpYyBTdGFuZGFyZCcsIGZpbGw6ICcjNjQ2NDY0JyB9KTtcbiAgICBjdXJyZW50VGltZVRleHQueCA9IHRoaXMuY2lyY2xlWCAtIGN1cnJlbnRUaW1lVGV4dC53aWR0aC8yO1xuICAgIGN1cnJlbnRUaW1lVGV4dC55ID0gdGhpcy5jaXJjbGVZIC0gY3VycmVudFRpbWVUZXh0LmhlaWdodC8yO1xuICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGN1cnJlbnRUaW1lVGV4dCk7XG4gICAgXG4gICAgc3RhZ2VWaWV3LnJlZ2lzdGVyQ29tcG9uZW50KHRoaXMpO1xuXG4gICAgLy9BZGQgQW5ub3RhdGlvbiB0byB0aGUgVGltZUxpbmVcbiAgICB0aGlzLmFkZEFubm90ID0gZnVuY3Rpb24oZGF0YSl7XG4gICAgXHRpZiAodHlwZW9mKHRoaXMuYW5ub3RDYXRlZ29yaWVzWzBdLmNvbG9yc1tkYXRhLmNvbnRlbnQuY2F0ZWdvcnkuY29kZV0pICE9PSAndW5kZWZpbmVkJyl7XG4gICAgXHRcdHZhciBhbm5vdENvZGUgPSBkYXRhLmNvbnRlbnQuY2F0ZWdvcnkuY29kZTtcbiAgICBcdH0gZWxzZSB7XG4gICAgXHRcdHZhciBhbm5vdENvZGUgPSB0aGlzLmFubm90Q2F0ZWdvcmllc1swXS5vcmRlclt0aGlzLmFubm90Q2F0ZWdvcmllc1swXS5vcmRlci5sZW5ndGggLTFdO1xuICAgIFx0fVxuICAgIFx0dmFyIGFubm90VGltZSA9IERhdGUucGFyc2UoZGF0YS50cyk7XG4gICAgXHRcbiAgICBcdGlmICh0aGlzLnRpbWVFbmQgPiBEYXRlLnBhcnNlKGRhdGEudHMpKXtcblx0ICAgIFx0dmFyIGkgPSBNYXRoLmZsb29yKChEYXRlLnBhcnNlKGRhdGEudHMpLXRoaXMudGltZUJlZ2luKS8oMTAwMCp0aGlzLmludGVydmFsRHVyYXRpb24pKTtcblx0ICAgIFx0XG5cdFx0XHR0aGlzLmNlbGxzW2ldLmNhdGVnb3JpZXNbYW5ub3RDb2RlXS5jb3VudCArPSAxO1xuXHRcdFx0dGhpcy5jZWxsc1tpXS50b3RhbEFubm90cyArPTE7XG5cdFx0XHR0aGlzLnJlZHJhd0NlbGwodGhpcy5jZWxsc1tpXSwgaSk7XG4gICAgXHR9XG4gICAgfTtcbiAgICBcbiAgICB0aGlzLmluaXRHcmFwaGljcyA9IGZ1bmN0aW9uKGNlbGwpe1xuICAgIFx0Y2VsbC5ncmFwaGljcyA9IG5ldyBQSVhJLkdyYXBoaWNzKCk7XG4gICAgXHRjZWxsLmdyYXBoaWNzLnBvc2l0aW9uLnggPSB0aGlzLmNpcmNsZVggKyB0aGlzLnJhZGl1cyAqIE1hdGguc2luKGNlbGwuaSooMzYwL3RvdGFsSW5kZXgpKihNYXRoLlBJLzE4MCkpO1xuICAgIFx0Y2VsbC5ncmFwaGljcy5wb3NpdGlvbi55ID0gdGhpcy5jaXJjbGVZIC0gdGhpcy5yYWRpdXMgKiBNYXRoLmNvcyhjZWxsLmkqKDM2MC90b3RhbEluZGV4KSooTWF0aC5QSS8xODApKTtcbiAgICBcdGNlbGwuZ3JhcGhpY3Mucm90YXRpb24gPSAoY2VsbC5pKSooMzYwL3RvdGFsSW5kZXgpKihNYXRoLlBJLzE4MCkgKyAoMzYwLyh0b3RhbEluZGV4KjIpKSooTWF0aC5QSS8xODApO1xuICAgIFx0dGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY2VsbC5ncmFwaGljcyk7XG4gICAgfVxuICAgIFxuICAgIHRoaXMuaW5pdFRpbWVUZXh0cyA9IGZ1bmN0aW9uKCkge1xuXHQgICAgdmFyIHRCZWcgPSBuZXcgUElYSS5UZXh0KFV0aWxzLmZvcm1hdFRpbWUodGhpcy50aW1lQmVnaW4pLCB7IGZvbnQ6ICcxMnB0IEdvdGhpYyBTdGFuZGFyZCcsIGZpbGw6ICcjNjQ2NDY0JyB9KTtcblx0ICAgIHRCZWcueCA9IHRoaXMuY2lyY2xlWCArIDE1O1xuXHQgICAgdEJlZy55ID0gdGhpcy5jaXJjbGVZIC0gdGhpcy5yYWRpdXMgLSB0aGlzLm1heENlbGxIZWlnaHQgLSAxMDtcblx0ICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKHRCZWcpO1xuXHQgICAgXG5cdCAgICB2YXIgdEVuZCA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSh0aGlzLnRpbWVFbmQpLCB7IGZvbnQ6ICcxMnB0IEdvdGhpYyBTdGFuZGFyZCcsIGZpbGw6ICcjNjQ2NDY0JyB9KTtcblx0ICAgIHRFbmQueCA9IHRoaXMuY2lyY2xlWCAtIDE1IC0gdEVuZC53aWR0aDtcblx0ICAgIHRFbmQueSA9IHRoaXMuY2lyY2xlWSAtIHRoaXMucmFkaXVzIC0gdGhpcy5tYXhDZWxsSGVpZ2h0IC0gMTA7XG5cdCAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZCh0RW5kKTtcblx0ICAgIFxuXHQgICAgdmFyIHQxNSA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSgoKHRoaXMudGltZUVuZCAtIHRoaXMudGltZUJlZ2luKS80KSArIHRoaXMudGltZUJlZ2luKSwgeyBmb250OiAnMTJwdCBHb3RoaWMgU3RhbmRhcmQnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG5cdCAgICB0MTUueCA9IHRoaXMuY2lyY2xlWCArIHRoaXMucmFkaXVzICsgdGhpcy5tYXhDZWxsSGVpZ2h0ICsgMTAgO1xuXHQgICAgdDE1LnkgPSB0aGlzLmNpcmNsZVkgLSB0MTUuaGVpZ2h0O1xuXHQgICAgdDE1LnJvdGF0aW9uID0gTWF0aC5QSSAvMjtcblx0ICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKHQxNSk7XG5cdCAgICBcblx0ICAgIHZhciB0MzAgPSBuZXcgUElYSS5UZXh0KFV0aWxzLmZvcm1hdFRpbWUoKCh0aGlzLnRpbWVFbmQgLSB0aGlzLnRpbWVCZWdpbikvMikgKyB0aGlzLnRpbWVCZWdpbiksIHsgZm9udDogJzEycHQgR290aGljIFN0YW5kYXJkJywgZmlsbDogJyM2NDY0NjQnIH0pO1xuXHQgICAgdDMwLnggPSB0aGlzLmNpcmNsZVggLSB0MzAud2lkdGgvMjtcblx0ICAgIHQzMC55ID0gdGhpcy5jaXJjbGVZICsgdGhpcy5yYWRpdXMgKyB0aGlzLm1heENlbGxIZWlnaHQgLSAyO1xuXHQgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodDMwKTtcblx0ICAgIFxuXHQgICAgdmFyIHQ0NSA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSgoKHRoaXMudGltZUVuZCAtIHRoaXMudGltZUJlZ2luKSozLzQpICsgdGhpcy50aW1lQmVnaW4pLCB7IGZvbnQ6ICcxMnB0IEdvdGhpYyBTdGFuZGFyZCcsIGZpbGw6ICcjNjQ2NDY0JyB9KTtcblx0ICAgIHQ0NS54ID0gdGhpcy5jaXJjbGVYIC0gdGhpcy5yYWRpdXMgLSB0aGlzLm1heENlbGxIZWlnaHQgLSAxMCA7XG5cdCAgICB0NDUueSA9IHRoaXMuY2lyY2xlWSArIHQxNS5oZWlnaHQ7XG5cdCAgICB0NDUucm90YXRpb24gPSAtTWF0aC5QSS8yO1xuXHQgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodDQ1KTtcbiAgICB9XG4gICAgXG4gICAgLy9EcmF3IHRoZSBjZWxsdWxlXG4gICAgdGhpcy5yZWRyYXdDZWxsID0gZnVuY3Rpb24oY2VsbCl7XG4gICAgXHRcbiAgICBcdGlmICh0eXBlb2YoY2VsbC5ncmFwaGljcykgPT09ICd1bmRlZmluZWQnKXtcbiAgICBcdFx0dGhpcy5pbml0R3JhcGhpY3MoY2VsbCk7XG4gICAgXHR9IGVsc2Uge1xuICAgIFx0XHRjZWxsLmdyYXBoaWNzLmNsZWFyKCk7XG4gICAgXHR9XG4gICAgXHRcbiAgICBcdHZhciB5ID0gMDtcbiAgICBcdFxuICAgIFx0Ly9DaGVjayBpZiB0b3RhbCBoZWlnaHQgaXMgaGlnaGVyIHRoYW4gTWF4IENlbGwgSGVpZ2h0XG4gICAgXHRpZiAoKGNlbGwudG90YWxBbm5vdHMqdGhpcy5pbnRlcnZhbEhlaWdodCkgPiB0aGlzLm1heENlbGxIZWlnaHQpe1xuICAgIFx0XHR2YXIgaGVpZ2h0U3RlcCA9IHRoaXMubWF4Q2VsbEhlaWdodC9jZWxsLnRvdGFsQW5ub3RzO1xuICAgIFx0fSBlbHNlIHtcbiAgICBcdFx0dmFyIGhlaWdodFN0ZXAgPSB0aGlzLmludGVydmFsSGVpZ2h0O1xuICAgIFx0fVxuICAgIFx0XG4gICAgXHQvL0RyYXcgdGhlIHJlY3QgZGVwZW5kaW5nIG9uIHRoZSBoZWlnaHQgc3RlcCBjYWxjdWxhdGVkXG4gICAgXHRmb3IgKHZhciBpPTA7IGk8IHRoaXMuYW5ub3RDYXRlZ29yaWVzWzBdLm9yZGVyLmxlbmd0aDsgaSsrKXtcbiAgICBcdFx0dmFyIGN1cnJlbnRDb2RlID0gdGhpcy5hbm5vdENhdGVnb3JpZXNbMF0ub3JkZXJbaV07XG5cdFx0XHRjZWxsLmdyYXBoaWNzLmJlZ2luRmlsbChjZWxsLmNhdGVnb3JpZXNbY3VycmVudENvZGVdLmNvbG9yLnJlcGxhY2UoXCIjXCIsIFwiMHhcIikpXG4gICAgXHRcdFx0LmRyYXdSZWN0KDAsIHksIHRoaXMuaW50ZXJ2YWxXaWR0aC0xLCAtY2VsbC5jYXRlZ29yaWVzW2N1cnJlbnRDb2RlXS5jb3VudCAqIGhlaWdodFN0ZXApXG4gICAgXHRcdFx0LmVuZEZpbGwoKTtcbiAgICBcdFx0eSAtPSBjZWxsLmNhdGVnb3JpZXNbY3VycmVudENvZGVdLmNvdW50KmhlaWdodFN0ZXA7XG4gICAgXHR9XG4gICAgfVxuICAgIFxuICAgIHRoaXMuaW5pdCA9IGZ1bmN0aW9uKCkge1xuICAgIFx0d3MubWVzc2FnZShmdW5jdGlvbihkYXRhKSB7XG4gICAgICAgICAgICBfdGhpcy5hZGRBbm5vdChkYXRhKTtcbiAgICAgICAgfSk7XG4gICAgXHRcbiAgICBcdHRoaXMuaW5pdFRpbWVUZXh0cygpO1xuICAgIH07XG4gICAgXG4gICAgdGhpcy51cGRhdGVUaW1lID0gZnVuY3Rpb24oKXtcbiAgICBcdGN1cnJlbnRUaW1lICs9IDEwMDA7XG4gICAgXHRcbiAgICAgICAgdmFyIG5iU2VjID0gY3VycmVudFRpbWUgLyAxMDAwO1xuICAgICAgICB2YXIgaG91cnMgPSBNYXRoLmZsb29yKCBuYlNlYyAvIDM2MDAgKSAlIDI0O1xuICAgICAgICB2YXIgbWludXRlcyA9IE1hdGguZmxvb3IoIG5iU2VjIC8gNjAgKSAlIDYwO1xuICAgICAgICB2YXIgc2Vjb25kcyA9IE1hdGguZmxvb3IobmJTZWMgJSA2MCk7XG4gICAgICAgIHZhciB0aW1lU3RyID0gKGhvdXJzIDwgMTAgPyAnMCcgKyBob3VycyA6IGhvdXJzKSArICc6JyArIChtaW51dGVzIDwgMTAgPyAnMCcgKyBtaW51dGVzIDogbWludXRlcykgKyAnOicgKyAoc2Vjb25kcyAgPCAxMCA/ICcwJyArIHNlY29uZHMgOiBzZWNvbmRzKTtcbiAgICAgICAgXG4gICAgICAgIGN1cnJlbnRUaW1lVGV4dC5zZXRUZXh0KHRpbWVTdHIpO1xuICAgIH07XG4gICAgXG4gICAgdmFyIHJlZnJlc2hUaW1lSW50ZXJ2YWw7XG4gICAgXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuICAgIFx0cmVmcmVzaFRpbWVJbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uKCkge190aGlzLnVwZGF0ZVRpbWUoKTt9LCAxMDAwKTtcbiAgICB9O1xuICAgIFxuICAgIHRoaXMucmVmcmVzaCA9IGZ1bmN0aW9uKCkge1xuICAgIFx0XG4gICAgfTtcbiAgICBcbiAgICB0aGlzLnN0b3AgPSBmdW5jdGlvbigpe1xuICAgIFx0Y29uc29sZS5sb2codGhpcy5jZWxscyk7XG4gICAgfTtcbiAgICBcbiAgICByZXR1cm4gdGhpcztcbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG5cdEFubm90c1RpbWVMaW5lOiBBbm5vdHNUaW1lTGluZVxufTtcbiIsIi8qKlxuKiBqcy9hbm5vdHN2aXp2aWV3LmpzXG4qXG4qIFRoaXMgaXMgdGhlIHN0YXJ0aW5nIHBvaW50IGZvciB5b3VyIGFwcGxpY2F0aW9uLlxuKiBUYWtlIGEgbG9vayBhdCBodHRwOi8vYnJvd3NlcmlmeS5vcmcvIGZvciBtb3JlIGluZm9cbiovXG5cbid1c2Ugc3RyaWN0JztcblxudmFyIFBJWEkgPSByZXF1aXJlKCdwaXhpJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xudmFyIERvdWJsZVJvbGwgPSByZXF1aXJlKCcuL2RvdWJsZXJvbGwuanMnKTtcbnZhciBBbm5vdHNUaW1lTGluZSA9IHJlcXVpcmUoJy4vYW5ub3RzdGltZWxpbmUuanMnKTtcbnZhciBBbm5vdHNSb2xsID0gcmVxdWlyZSgnLi9hbm5vdHNyb2xsLmpzJyk7XG5cbnZhciBkZWZhdWx0T3B0aW9ucyA9IHtcbiAgICB4SW5pdDogMCxcbiAgICB5SW5pdDogMCxcbiAgICB3aWR0aDogMTAyNCxcbiAgICBoZWlnaHQ6IDc2OCxcbiAgICBhbm5vdENhdGVnb3JpZXM6IFt7XG4gICAgICAgIHRzOiAwLFxuICAgICAgICBjb2xvcnM6IHtcbiAgICAgICAgICAgICdudG0nOiAnI0NEQzgzRicsXG4gICAgICAgICAgICAnaWFtJzogJyNDREM4M0YnLFxuICAgICAgICAgICAgJ2hpcCc6ICcjQ0RDODNGJyxcbiAgICAgICAgICAgICdob3AnOiAnI0NEQzgzRicsXG4gICAgICAgICAgICAncm9jayc6ICcjREU4QjUzJyxcbiAgICAgICAgICAgICdyYXAnOiAnI0RFOEI1MycsXG4gICAgICAgICAgICAnY2xhc3NpYyc6ICcjREU4QjUzJyxcbiAgICAgICAgICAgICdkcnVtcyc6ICcjQzVBM0NBJyxcbiAgICAgICAgICAgICdndWl0YXInOiAnI0M1QTNDQScsXG4gICAgICAgICAgICAnYmFzcyc6ICcjNzlCQjkyJyxcbiAgICAgICAgICAgICdkZWZhdWx0JzogJyM4MDgwODAnXG4gICAgICAgIH0sXG4gICAgICAgIG9yZGVyOiBbJ250bScsICdpYW0nLCAnaGlwJywgJ2hvcCcsICdyb2NrJywgJ3JhcCcsICdjbGFzc2ljJywgJ2RydW1zJywgJ2d1aXRhcicsICdiYXNzJywgJ2RlZmF1bHQnXVxuICAgIH1dXG59O1xuXG5mdW5jdGlvbiBBbm5vdHNWaXpWaWV3KG9wdGlvbnMpe1xuXHR2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0T3B0aW9ucykudmFsdWUoKTtcblxuICAgIHRoaXMuY29udGFpbmVyID0gbmV3IFBJWEkuRGlzcGxheU9iamVjdENvbnRhaW5lcigpO1xuICAgIHRoaXMuY29udGFpbmVyLnggPSBvcHRzLnhJbml0O1xuICAgIHRoaXMuY29udGFpbmVyLnkgPSBvcHRzLnlJbml0O1xuXHR0aGlzLndpZHRoID0gb3B0cy53aWR0aDtcblx0dGhpcy5oZWlnaHQ9IG9wdHMuaGVpZ2h0O1xuICAgIHRoaXMuYW5ub3RDYXRlZ29yaWVzID0gb3B0cy5hbm5vdENhdGVnb3JpZXM7XG5cblx0dmFyIHdzUGlhbm9yb2xsID0gb3B0cy53c1BpYW5vcm9sbDtcblx0dmFyIHdzQW5ub3QgPSBvcHRzLndzQW5ub3Q7XG5cdHZhciBzdGFnZVZpZXcgPSBvcHRzLnN0YWdlVmlldztcblxuXHRzdGFnZVZpZXcucmVnaXN0ZXJDb21wb25lbnQodGhpcyk7XG5cblx0dmFyIHRpbWVMaW5lID0gbmV3IEFubm90c1RpbWVMaW5lLkFubm90c1RpbWVMaW5lKHtcbiAgICBcdHN0YWdlVmlldyA6IHN0YWdlVmlldyxcbiAgICAgICAgbG9nZ2VyOiBsb2dnZXIsXG4gICAgICAgIHdzOiBuZXcgYW5ub3R2aXouV3NXcmFwcGVyKHdzVXJpQW5ub3RhdGlvbiwgbG9nZ2VyKSxcbiAgICAgICAgeEluaXQ6IDAsXG4gICAgICAgIHlJbml0OiAwLFxuICAgICAgICB3aWR0aDogMTAyNCAtIDIwMCAtIDIwMCxcbiAgICAgICAgaGVpZ2h0OiA3NjgtMjAwLFxuICAgICAgICB0aW1lQmVnaW46IERhdGUubm93KCksXG4gICAgICAgIHRpbWVFbmQ6IERhdGUubm93KCkgKyAzMDAwMDAwLFxuICAgICAgICBpbnRlcnZhbFdpZHRoOiA2LFxuICAgICAgICBpbnRlcnZhbEhlaWdodDogMTAsXG4gICAgICAgIG1heENlbGxIZWlnaHQ6IDcwLFxuICAgICAgICByYWRpdXM6IDIwMCxcbiAgICAgICAgYW5ub3RDYXRlZ29yaWVzOiB0aGlzLmFubm90Q2F0ZWdvcmllc1xuICAgIH0pO1xuXG5cdHZhciBkb3VibGVSb2xsSCA9IG5ldyBEb3VibGVSb2xsLkRvdWJsZVJvbGwoe1xuICAgICAgICBzdGFnZVZpZXcgOiBzdGFnZVZpZXcsXG4gICAgXHRsb2dnZXI6IGxvZ2dlcixcbiAgICAgICAgd3M6IHdzUGlhbm9yb2xsLFxuICAgICAgICB5SW5pdDogKHRoaXMuaGVpZ2h0IC0gMjAwKSxcbiAgICAgICAgc2NlbmVIZWlnaHQ6IDIwMCxcbiAgICAgICAgcGlhbm9yb2xscyA6IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICBoZWlnaHQ6IDIwMCxcbiAgICAgICAgICAgICAgICB0aW1lV2lkdGg6IDEwLFxuICAgICAgICAgICAgICAgIGxpbmVJbnRlcnZhbDogNTAwMCxcbiAgICAgICAgICAgICAgICBub3RlSGVpZ2h0OiAxMFxuICAgICAgICAgICAgfSxcbiAgICAgICAgXVxuICAgIH0pO1xuXG5cdHZhciBkb3VibGVSb2xsViA9IG5ldyBEb3VibGVSb2xsLkRvdWJsZVJvbGwoe1xuICAgICAgICBzdGFnZVZpZXcgOiBzdGFnZVZpZXcsXG4gICAgXHRsb2dnZXI6IGxvZ2dlcixcbiAgICAgICAgd3M6IHdzUGlhbm9yb2xsLFxuICAgICAgICBvcmllbnRhdGlvbjogJ3ZlcnRpY2FsJyxcbiAgICAgICAgc2NlbmVIZWlnaHQ6IDc2OC0yMDAsXG4gICAgICAgIHBpYW5vcm9sbHMgOiBbXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgaGVpZ2h0OiAyMDAsXG4gICAgICAgICAgICAgICAgdGltZVdpZHRoOiA2MCxcbiAgICAgICAgICAgICAgICBsaW5lSW50ZXJ2YWw6IDUwMDAsXG4gICAgICAgICAgICAgICAgbm90ZUhlaWdodDogNSxcbiAgICAgICAgICAgIH0sXG4gICAgICAgIF1cbiAgICB9KTtcblxuXHR2YXIgYW5ub3RzUm9sbCA9IG5ldyBBbm5vdHNSb2xsLkFubm90c1JvbGwoe1xuICAgIFx0c3RhZ2VWaWV3IDogc3RhZ2VWaWV3LFxuICAgICAgICBsb2dnZXI6IGxvZ2dlcixcbiAgICAgICAgd3M6IHdzQW5ub3QsXG4gICAgICAgIHBhcmVudENvbnRhaW5lcjogZG91YmxlUm9sbFYuc3RhZ2UsXG4gICAgICAgIHhJbml0OiAxMDI0IC0gMjAwIC0gMjAwLFxuICAgICAgICB5SW5pdDogNzY4LTIwMCxcbiAgICAgICAgd2lkdGg6IDIwMCArIDIwMCxcbiAgICAgICAgaGVpZ2h0OiA3NjgtMjAwLFxuICAgICAgICB3aWR0aFJvbGw6IDIwMCxcbiAgICAgICAgZnJhbWVyYXRlOiBkb3VibGVSb2xsVi5mcmFtZXJhdGUsXG4gICAgICAgIHBpeGVsc1BlclNlY29uZDogTWF0aC5mbG9vcigxMDI0IC8gNjApLFxuICAgICAgICBhbm5vdENvbG9yczogdGhpcy5hbm5vdENhdGVnb3JpZXNcbiAgICB9KTtcblxuXHR2YXIgbGltaXRlcnMgPSBuZXcgUElYSS5HcmFwaGljcygpXG5cdFx0LmxpbmVTdHlsZSgxLCAweDY0NjQ2NClcblx0XHQubW92ZVRvKGFubm90c1JvbGwuY29udGFpbmVyLngsIGFubm90c1JvbGwuY29udGFpbmVyLnkpXG5cdFx0LmxpbmVUbyhhbm5vdHNSb2xsLmNvbnRhaW5lci54LCBhbm5vdHNSb2xsLmNvbnRhaW5lci55IC0gYW5ub3RzUm9sbC5oZWlnaHQpXG5cdFx0Lm1vdmVUbyhhbm5vdHNSb2xsLmNvbnRhaW5lci54ICsgYW5ub3RzUm9sbC53aWR0aFJvbGwsIGFubm90c1JvbGwuY29udGFpbmVyLnkpXG5cdFx0LmxpbmVUbyhhbm5vdHNSb2xsLmNvbnRhaW5lci54ICsgYW5ub3RzUm9sbC53aWR0aFJvbGwsIGFubm90c1JvbGwuY29udGFpbmVyLnkgLSBhbm5vdHNSb2xsLmhlaWdodClcblx0XHQubW92ZVRvKDAsIHRoaXMuaGVpZ2h0IC0gMjAwKVxuXHRcdC5saW5lVG8odGhpcy53aWR0aCwgdGhpcy5oZWlnaHQgLSAyMDApXG5cdFx0LmRyYXdSZWN0KDAsIDAsIHRoaXMud2lkdGggLTEsIHRoaXMuaGVpZ2h0IC0xKVxuXHRcdC5iZWdpbkZpbGwoMHhFQ0VDRUMpXG5cdFx0LmRyYXdSZWN0KDEwMjQgLSAyMDAsIDAsIDIwMCwgNzY4LTIwMClcblx0XHQuZW5kRmlsbCgpO1xuXHR0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChsaW1pdGVycyk7XG5cbi8vXHR2YXIgZG91YmxlUm9sbFYgPSBuZXcgRG91YmxlUm9sbC5Eb3VibGVSb2xsKHt9KTtcblxuXHR0aGlzLmluaXQgPSBmdW5jdGlvbigpe1xuXG5cdH1cblxuXHR0aGlzLnN0YXJ0ID0gZnVuY3Rpb24oKSB7XG4gICAgfTtcblxuICAgIHRoaXMucmVmcmVzaCA9IGZ1bmN0aW9uKCkge1xuICAgIH07XG5cbiAgICB0aGlzLnN0b3AgPSBmdW5jdGlvbigpe1xuICAgIH07XG5cbiAgICByZXR1cm4gdGhpcztcblxufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcblx0QW5ub3RzVml6VmlldzogQW5ub3RzVml6Vmlld1xufTtcbiIsIi8qKlxuKiBzY3JpcHRzL2RvdWJsZXJvbGwuanNcbipcbiogVGhpcyBpcyB0aGUgc3RhcnRpbmcgcG9pbnQgZm9yIHlvdXIgYXBwbGljYXRpb24uXG4qIFRha2UgYSBsb29rIGF0IGh0dHA6Ly9icm93c2VyaWZ5Lm9yZy8gZm9yIG1vcmUgaW5mb1xuKi9cblxuLyogZ2xvYmFsIGRvY3VtZW50OiBmYWxzZSAqL1xuXG4ndXNlIHN0cmljdCc7XG5cblxudmFyIFBJWEkgPSByZXF1aXJlKCdwaXhpJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xudmFyIFBpYW5vUm9sbCA9IHJlcXVpcmUoJy4vcGlhbm9yb2xsLmpzJyk7XG5cbnZhciBkZWZhdWx0Q29uZmlnID0ge1xuICAgIG9yaWVudGF0aW9uOiAnaG9yaXpvbnRhbCcsXG4gICAgbG9nZ2VyOiB1bmRlZmluZWQsXG4gICAgc2NlbmVXaWR0aDogMTAyNCxcbiAgICBwaWFub3JvbGxzIDogW1xuICAgICAge1xuICAgICAgICBoZWlnaHQ6IDQzNSxcbiAgICAgICAgdGltZVdpZHRoOiAxMCxcbiAgICAgICAgbGluZUludGVydmFsOiA1MDAwLFxuICAgICAgICBub3RlSGVpZ2h0OiB1bmRlZmluZWRcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIGhlaWdodDogNjQ1LFxuICAgICAgICB0aW1lV2lkdGg6IDYwLFxuICAgICAgICBsaW5lSW50ZXJ2YWw6IDUwMDAsXG4gICAgICAgIG5vdGVIZWlnaHQ6IHVuZGVmaW5lZFxuICAgICAgfSxcbiAgICBdLFxuICAgIGZyYW1lcmF0ZTogMjUsXG4gICAgb2Zmc2V0TXVzaWM6IGZhbHNlLFxuICAgIHNjZW5lQmdDb2xvcjogMHhGRkZGRkYsXG4gICAgbGluZUNvbG9yOiAweDQ0NDQ0NCxcbiAgICBsaW5lRmlsbENvbG9yOiAweEZGRkYwMCxcbiAgICBub3RlQ29sb3JzOiBbMHhCOTAwMDAsIDB4NEJERDcxLCAweEFGOTMxRSwgMHgxQzI4QkEsIDB4NTM2OTkxXSxcbiAgICBub3RlSGVpZ2h0OiB1bmRlZmluZWQsXG4gICAgemVyb1NoaWZ0OiAwLjksXG4gICAgdGltZVdpZHRoOiA2MCxcbiAgICBsaW5lSW50ZXJ2YWw6IDUwMDAsXG4vLyAgICB3c1VyaTogdW5kZWZpbmVkLFxuLy8gICAgZXZlbnRDb2RlOiB1bmRlZmluZWRcblxufTtcblxuZnVuY3Rpb24gRG91YmxlUm9sbChvcHRpb25zKSB7XG5cbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0Q29uZmlnKS52YWx1ZSgpO1xuXG4gICAgdmFyIG9yaWVudGF0aW9uID0gb3B0cy5vcmllbnRhdGlvbjtcbiAgICB2YXIgaXNIb3Jpem9udGFsID0gKG9yaWVudGF0aW9uICE9PSAndmVydGljYWwnKTtcblxuICAgIHRoaXMubG9nZ2VyID0gb3B0cy5sb2dnZXI7XG4gICAgdGhpcy5saW5lQ29sb3IgPSBvcHRzLmxpbmVDb2xvcjtcbiAgICB0aGlzLmxpbmVGaWxsQ29sb3IgPSBvcHRzLmxpbmVGaWxsQ29sb3I7XG4gICAgdGhpcy5mcmFtZXJhdGUgPSBvcHRzLmZyYW1lcmF0ZTtcbiAgICB0aGlzLm9mZnNldE11c2ljID0gb3B0cy5vZmZzZXRNdXNpYztcbiAgICB0aGlzLm5vdGVDb2xvcnMgPSBvcHRzLm5vdGVDb2xvcnM7XG5cbiAgICB2YXIgbm90ZUhlaWdodCA9IG9wdHMubm90ZUhlaWdodDtcbiAgICB2YXIgc2NlbmVCZ0NvbG9yID0gb3B0cy5zY2VuZUJnQ29sb3I7XG4gICAgdmFyIHNjZW5lSGVpZ2h0ID0gb3B0cy5zY2VuZUhlaWdodCB8fCBfKG9wdHMucGlhbm9yb2xscykucmVkdWNlKGZ1bmN0aW9uKHMscCkgeyByZXR1cm4gcyArIHAuaGVpZ2h0OyB9LCAwKTtcbiAgICB2YXIgdGltZVdpZHRoID0gb3B0cy50aW1lV2lkdGg7XG4gICAgdmFyIGxpbmVJbnRlcnZhbCA9IG9wdHMubGluZUludGVydmFsO1xuICAgIHZhciBvZmZzZXRNdXNpYyA9IG9wdHMub2Zmc2V0TXVzaWM7XG5cbiAgICB2YXIgc2NlbmVXaWR0aCA9IG9wdHMuc2NlbmVXaWR0aDtcbiAgICB2YXIgc3RhZ2VWaWV3ID0gb3B0cy5zdGFnZVZpZXc7XG5cbiAgICB2YXIgemVyb1NoaWZ0ID0gb3B0cy56ZXJvU2hpZnQ7XG5cbiAgICB2YXIgd3MgPSBvcHRzLndzO1xuXG4gICAgdmFyIGNvbG9yc1JlZyA9IHt9O1xuXG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IE1hdGguZmxvb3Ioc2NlbmVXaWR0aCp6ZXJvU2hpZnQpO1xuICAgIHRoaXMuY29udGFpbmVyLnkgPSAwO1xuICAgIFxuICAgIHN0YWdlVmlldy5yZWdpc3RlckNvbXBvbmVudCh0aGlzKTtcblxuICAgIHZhciBwaWFub3JvbGxMaXN0ID0gW107XG5cbiAgICB2YXIgcGlhbm9yb2xsT3B0aW9ucyA9IHtcbiAgICAgICAgcGFyZW50Q29udGFpbmVyOiB0aGlzLmNvbnRhaW5lcixcbiAgICAgICAgb3JpZW50YXRpb246IG9yaWVudGF0aW9uLFxuICAgICAgICB4SW5pdDogMCxcbiAgICAgICAgd2lkdGg6IHNjZW5lV2lkdGgsXG4gICAgICAgIG5vdGVDb2xvcnM6IHRoaXMubm90ZUNvbG9ycyxcbiAgICAgICAgY29sb3JzUmVnOiBjb2xvcnNSZWcsXG4gICAgICAgIGxpbmVDb2xvcjogdGhpcy5saW5lQ29sb3IsXG4gICAgICAgIGxpbmVJbnRlcnZhbDogbGluZUludGVydmFsLFxuICAgICAgICBvZmZzZXRNdXNpYzogb2Zmc2V0TXVzaWMsXG4gICAgfTtcblxuICAgIHZhciB5SW5pdCA9IG9wdHMueUluaXQgfHwgMDtcbiAgICB2YXIgbGluZXNEb3duID0gdHJ1ZTtcbiAgICBfKG9wdHMucGlhbm9yb2xscykuZm9yRWFjaChmdW5jdGlvbihwckRlZiwgaSkge1xuICAgICAgICB2YXIgcHJOb3RlSGVpZ2h0ID0gbm90ZUhlaWdodCB8fCBwckRlZi5ub3RlSGVpZ2h0IHx8IHByRGVmLmhlaWdodCAvIDEyODtcbiAgICAgICAgdmFyIHByVGltZVdpZHRoID0gcHJEZWYudGltZVdpZHRoIHx8IHRpbWVXaWR0aDtcbiAgICAgICAgcGlhbm9yb2xsTGlzdC5wdXNoKG5ldyBQaWFub1JvbGwoXyh7XG4gICAgICAgICAgICB5SW5pdDogeUluaXQsXG4gICAgICAgICAgICBoZWlnaHQ6IHByRGVmLmhlaWdodCxcbiAgICAgICAgICAgIGxpbmVzRG93bjogbGluZXNEb3duLFxuICAgICAgICAgICAgcGl4ZWxzUGVyU2Vjb25kOiBNYXRoLmZsb29yKHNjZW5lV2lkdGggLyBwclRpbWVXaWR0aCksXG4gICAgICAgICAgICBub3RlSGVpZ2h0OiBwck5vdGVIZWlnaHQsXG4gICAgICAgICAgICBsaW5lSW50ZXJ2YWw6IHByRGVmLmxpbmVJbnRlcnZhbFxuICAgICAgICB9KS5kZWZhdWx0cyhwaWFub3JvbGxPcHRpb25zKS52YWx1ZSgpKSk7XG4gICAgICAgIHlJbml0ICs9IHByRGVmLmhlaWdodDtcbiAgICAgICAgbGluZXNEb3duID0gIWxpbmVzRG93bjtcblxuICAgICAgICBpZihpPChvcHRzLnBpYW5vcm9sbHMubGVuZ3RoLTEpKSB7XG4gICAgICAgICAgICB2YXIgbGluZUdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKVxuICAgICAgICAgICAgICAgIC5iZWdpbkZpbGwoX3RoaXMubGluZUZpbGxDb2xvcilcbiAgICAgICAgICAgICAgICAubGluZVN0eWxlKDEsIF90aGlzLmxpbmVDb2xvcilcbiAgICAgICAgICAgICAgICAubW92ZVRvKE1hdGguZmxvb3Ioc2NlbmVXaWR0aCp6ZXJvU2hpZnQpLCB5SW5pdClcbiAgICAgICAgICAgICAgICAubGluZVRvKC1zY2VuZVdpZHRoIC0gTWF0aC5mbG9vcihzY2VuZVdpZHRoKnplcm9TaGlmdCksIHlJbml0KVxuICAgICAgICAgICAgICAgIC5lbmRGaWxsKCk7XG4gICAgICAgICAgICBfdGhpcy5jb250YWluZXIuYWRkQ2hpbGQobGluZUdyYXBoaWNzKTtcbiAgICAgICAgfVxuICAgIH0pO1xuXG4gICAgaWYoIWlzSG9yaXpvbnRhbCkge1xuICAgICAgICB0aGlzLmNvbnRhaW5lci5yb3RhdGlvbiA9IE1hdGguUEkvMjtcbiAgICAgICAgdGhpcy5jb250YWluZXIueSA9IHNjZW5lSGVpZ2h0O1xuICAgICAgICB0aGlzLmNvbnRhaW5lci54ID0gc2NlbmVXaWR0aDtcbiAgICB9XG5cblxuICAgIHRoaXMuaW5pdCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgXHR3cy5tZXNzYWdlKGZ1bmN0aW9uKGRhdGEpIHtcbiAgICAgICAgICAgIF90aGlzLmFkZE5vdGVzKGRhdGEpO1xuICAgICAgICB9KTtcblxuICAgIH07XG5cblxuICAgIHRoaXMuYWRkTm90ZXMgPSBmdW5jdGlvbihkYXRhKSB7XG5cbiAgICAgICAgcGlhbm9yb2xsTGlzdC5mb3JFYWNoKGZ1bmN0aW9uKGMpIHtcbiAgICAgICAgICAgIGMuYWRkTm90ZVJhdyhkYXRhKTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIHRoaXMucmVmcmVzaCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBwaWFub3JvbGxMaXN0LmZvckVhY2goZnVuY3Rpb24oYykge1xuICAgICAgICAgICAgYy5tb3ZlKCk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cbiAgICAvLyBJbml0IHBhZ2UgYW5kIGludGVydmFsc1xuICAgIHZhciBzdGFydFRzO1xuXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgICAgIHN0YXJ0VHMgPSBEYXRlLm5vdygpO1xuICAgICAgICBwaWFub3JvbGxMaXN0LmZvckVhY2goZnVuY3Rpb24oYykge1xuICAgICAgICAgICAgYy5zdGFydCgpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5zdG9wID0gZnVuY3Rpb24oKSB7XG4gICAgXHRcbiAgICAgICAgcGlhbm9yb2xsTGlzdC5mb3JFYWNoKGZ1bmN0aW9uKGMpIHtcbiAgICAgICAgICAgIGMuc3RvcCgpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG5cbiAgICB0aGlzLmxvZyA9IGZ1bmN0aW9uKG0pIHtcbiAgICAgICAgaWYodGhpcy5sb2dnZXIpIHtcbiAgICAgICAgICAgIHRoaXMubG9nZ2VyLmxvZyhtKTtcbiAgICAgICAgfVxuICAgIH07XG5cblxuXG4gICAgcmV0dXJuIHRoaXM7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIERvdWJsZVJvbGw6IERvdWJsZVJvbGxcbn07XG4iLCIvKipcbioganMvd3N3cmFwcGVyLmpzXG4qXG4qIHNpbXBsZSBsb2dnZXIgc2VydmljZVxuKlxuKi9cblxuLyogZ2xvYmFsIGRvY3VtZW50OiBmYWxzZSAqL1xuXG4ndXNlIHN0cmljdCc7XG5cbmZ1bmN0aW9uIEh0bWxMb2dnZXIoZG9Mb2csIGNvbnRhaW5lcikge1xuXG4gICAgdmFyIGxvZ0NvbnRhaW5lciA9IGNvbnRhaW5lcjtcbiAgICBpZih0eXBlb2YoY29udGFpbmVyKSA9PT0gJ3N0cmluZycpIHtcbiAgICAgICAgbG9nQ29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoY29udGFpbmVyKTtcbiAgICB9XG4gICAgaWYoIWRvTG9nKSB7XG4gICAgICAgIGRvY3VtZW50LmJvZHkucmVtb3ZlQ2hpbGQobG9nQ29udGFpbmVyKTtcbiAgICAgICAgbG9nQ29udGFpbmVyID0gdW5kZWZpbmVkO1xuICAgIH1cblxuXG4gICAgdGhpcy5sb2cgPSBmdW5jdGlvbihtc2cpIHtcbiAgICAgICAgaWYoZG9Mb2cgJiYgbG9nQ29udGFpbmVyKSB7XG4gICAgICAgICAgICBsb2dDb250YWluZXIuaW5uZXJIVE1MICs9IG1zZyArICdcXG4nO1xuICAgICAgICAgICAgbG9nQ29udGFpbmVyLnNjcm9sbFRvcCA9IGxvZ0NvbnRhaW5lci5zY3JvbGxIZWlnaHQ7XG4gICAgICAgIH1cbiAgICB9O1xufVxuXG5mdW5jdGlvbiBDb25zb2xlTG9nZ2VyKGRvTG9nKSB7XG5cbiAgICB0aGlzLmxvZyA9IGZ1bmN0aW9uKG1zZykge1xuICAgICAgICBpZihkb0xvZykge1xuICAgICAgICAgICAgY29uc29sZS5sb2cobXNnKTtcbiAgICAgICAgfVxuICAgIH1cblxufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBIdG1sTG9nZ2VyOiBIdG1sTG9nZ2VyLFxuICAgIENvbnNvbGVMb2dnZXI6IENvbnNvbGVMb2dnZXJcbn07XG4iLCIvKipcbioganMvcGlhbm9yb2xsLmpzXG4qXG4qIHBpYW5vcm9sbCBiYXNpYyBjb21wb25lbnRcbipcbiovXG5cbid1c2Ugc3RyaWN0JztcblxuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciByYW5kb21Db2xvciA9IHJlcXVpcmUoJ3JhbmRvbUNvbG9yJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xuXG52YXIgTlRQX0VQT0NIX0RFTFRBID0gMjIwODk4ODgwMDsgLy9jLmYuIFJGQyA4NjhcblxuZnVuY3Rpb24gUGlhbm9Sb2xsKG9wdGlvbnMpIHtcbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHRoaXMuY29udGFpbmVyID0gbmV3IFBJWEkuRGlzcGxheU9iamVjdENvbnRhaW5lcigpO1xuICAgIHRoaXMuY29udGFpbmVyLnggPSBvcHRpb25zLnhJbml0O1xuICAgIHRoaXMuY29udGFpbmVyLnkgPSBvcHRpb25zLnlJbml0O1xuICAgIG9wdGlvbnMucGFyZW50Q29udGFpbmVyLmFkZENoaWxkKHRoaXMuY29udGFpbmVyKTtcblxuICAgIHZhciBvcmllbnRhdGlvbiA9IG9wdGlvbnMub3JpZW50YXRpb247XG4gICAgdmFyIGlzSG9yaXpvbnRhbCA9IChvcmllbnRhdGlvbiAhPT0gJ3ZlcnRpY2FsJyk7XG5cbiAgICB0aGlzLmxpbmVzRG93biA9IG9wdGlvbnMubGluZXNEb3duO1xuICAgIHRoaXMuaGVpZ2h0ID0gb3B0aW9ucy5oZWlnaHQ7XG4gICAgdGhpcy5waXhlbHNQZXJTZWNvbmQgPSBvcHRpb25zLnBpeGVsc1BlclNlY29uZDtcbiAgICB0aGlzLndpZHRoID0gb3B0aW9ucy53aWR0aDtcbiAgICB0aGlzLm5vdGVDb2xvcnMgPSBvcHRpb25zLm5vdGVDb2xvcnM7XG4gICAgdGhpcy5jb2xvcnNSZWcgPSBvcHRpb25zLmNvbG9yc1JlZyB8fCB7fTtcbiAgICB0aGlzLmxpbmVDb2xvciA9IG9wdGlvbnMubGluZUNvbG9yO1xuICAgIHRoaXMubGluZUludGVydmFsID0gb3B0aW9ucy5saW5lSW50ZXJ2YWw7XG4gICAgdGhpcy5vZmZzZXRNdXNpYyA9IG9wdGlvbnMub2Zmc2V0TXVzaWMgfHwgZmFsc2U7XG4gICAgdGhpcy5ub3RlSGVpZ2h0ID0gb3B0aW9ucy5ub3RlSGVpZ2h0O1xuICAgIHRoaXMubm90ZURpY3QgPSB7fTtcbiAgICB0aGlzLnN0YXJ0VHMgPSBvcHRpb25zLnN0YXJ0VHMgfHwgRGF0ZS5ub3coKTtcblxuICAgIHZhciBzdGFydGVkID0gZmFsc2U7XG5cbiAgICB2YXIgaXNIaWRkZW4gPSBmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICAvLyBUT0RPOiB0aGUgb3JpZ2luIHBvaW50IGlzIGFuIGFwcHJveGltYXRpb24uIFNob3VsZCByZWZpbmUgdGhpc1xuICAgICAgICB2YXIgZ2xvYmFsUG9zID0gY2hpbGQudG9HbG9iYWwobmV3IFBJWEkuUG9pbnQoMCwwKSk7XG4gICAgICAgIHJldHVybiAoKGdsb2JhbFBvcy54ICsgY2hpbGQud2lkdGgpIDwgMCkgfHwgKChnbG9iYWxQb3MueSArIGNoaWxkLmhlaWdodCkgPCAwKSA7XG4gICAgfTtcblxuICAgIC8vVE9ETzogSSBkbyBub3QgbGlrZSB0aGUgXCJyZWdDb2xvclwiIG9iamVjdC4gVGhpcyBzaG91bGQgbm90IGJlIGdsb2JhbCwgYnV0IGxvY2FsXG4gICAgdGhpcy5nZXRDb2xvciA9IGZ1bmN0aW9uKGNhbmFsKSB7XG4gICAgICAgIHZhciBjb2xvciA9IHRoaXMuY29sb3JzUmVnW2NhbmFsXTtcbiAgICAgICAgaWYodHlwZW9mKGNvbG9yKSA9PT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgICAgICAgIHZhciBjb2xvcnNSZWdTaXplID0gT2JqZWN0LmtleXModGhpcy5jb2xvcnNSZWcpLmxlbmd0aDtcbiAgICAgICAgICAgIGlmKGNvbG9yc1JlZ1NpemUgPCB0aGlzLm5vdGVDb2xvcnMubGVuZ3RoKSB7XG4gICAgICAgICAgICAgICAgY29sb3IgPSB0aGlzLmNvbG9yc1JlZ1tjYW5hbF0gPSB0aGlzLm5vdGVDb2xvcnNbY29sb3JzUmVnU2l6ZV07XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBjb2xvciA9IHRoaXMuY29sb3JzUmVnW2NhbmFsXSA9IHBhcnNlSW50KHJhbmRvbUNvbG9yKHsgbHVtaW5vc2l0eTogJ2xpZ2h0JywgaHVlOiAncmFuZG9tJywgZm9ybWF0OidoZXgnfSkucmVwbGFjZSgvXiMvLCAnJyksIDE2KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gY29sb3I7XG4gICAgfTtcblxuICAgIHRoaXMuZ2V0Tm90ZVJlY3QgPSBmdW5jdGlvbih4LCB5LCBjb2xvciwgYWxwaGEsIHdpZHRoLCBoZWlnaHQpIHtcbiAgICAgICAgdmFyIGdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKTtcbiAgICAgICAgZ3JhcGhpY3MuYmVnaW5GaWxsKGNvbG9yLCBhbHBoYSk7XG4gICAgICAgIGdyYXBoaWNzLmRyYXdSZWN0KDAsIDAsIHdpZHRoLCBoZWlnaHQpO1xuICAgICAgICBncmFwaGljcy5lbmRGaWxsKCk7XG4gICAgICAgIGdyYXBoaWNzLnggPSB4O1xuICAgICAgICBncmFwaGljcy55ID0geTtcbiAgICAgICAgZ3JhcGhpY3Mud2lkdGggPSB3aWR0aDtcbiAgICAgICAgZ3JhcGhpY3MuaGVpZ2h0ID0gaGVpZ2h0O1xuICAgICAgICByZXR1cm4gZ3JhcGhpY3M7XG4gICAgfTtcblxuICAgIHRoaXMuYWRkTm90ZVJhdyA9IGZ1bmN0aW9uKGRhdGEpIHtcbiAgICBcdGNvbnNvbGUubG9nKGRhdGEpO1xuICAgICAgICB2YXIgbm90ZSA9IGRhdGEuY29udGVudFszXTtcbiAgICAgICAgdmFyIHZlbG9jaXR5ID0gZGF0YS5jb250ZW50WzRdO1xuICAgICAgICB2YXIgdHMgPSAoZGF0YS5jb250ZW50WzBdIC0gTlRQX0VQT0NIX0RFTFRBKSoxMDAwO1xuICAgICAgICB2YXIgY2hhbm5lbCA9IGRhdGEuY29udGVudFsyXTtcbiAgICAgICAgdmFyIHNlc3Npb25UcyA9IGRhdGEuY29udGVudFsxXTtcblxuICAgICAgICB0aGlzLmFkZE5vdGUobm90ZSwgdHMsIHNlc3Npb25UcywgdmVsb2NpdHksIGNoYW5uZWwsIDApO1xuICAgIH07XG5cbiAgICB0aGlzLmFkZE5vdGUgPSBmdW5jdGlvbihub3RlLCBzdGFydFRpbWUsIHNlc3Npb25UcywgdmVsb2NpdHksIGNoYW5uZWwsIGR1cmF0aW9uKSB7XG5cbiAgICAgICAgdmFyIHRzID0gc3RhcnRUaW1lO1xuICAgICAgICBpZih0aGlzLm9mZnNldE11c2ljKSB7XG4gICAgICAgICAgICB0cyA9IHRoaXMuc3RhcnRUcyArIHNlc3Npb25UcztcbiAgICAgICAgfVxuXG4gICAgICAgIHZhciBub3RlRHVyYXRpb24gPSBkdXJhdGlvbjtcbiAgICAgICAgdmFyIG5vdGVWZWxvY2l0eSA9IHZlbG9jaXR5O1xuICAgICAgICB2YXIgZ3JhcGhpY3M7XG4gICAgICAgIGlmKCFkdXJhdGlvbikge1xuICAgICAgICAgICAgaWYodHlwZW9mIHRoaXMubm90ZURpY3RbY2hhbm5lbF09PT0ndW5kZWZpbmVkJyl7XG4gICAgICAgICAgICAgICAgdGhpcy5ub3RlRGljdFtjaGFubmVsXSA9IHt9O1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaWYodmVsb2NpdHk9PT0wKSB7XG4gICAgICAgICAgICAgICAgaWYodHlwZW9mIHRoaXMubm90ZURpY3RbY2hhbm5lbF1bbm90ZV0gIT09ICd1bmRlZmluZWQnKSB7XG4gICAgICAgICAgICAgICAgICAgIHZhciBub3RlRGVmID0gdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXTtcbiAgICAgICAgICAgICAgICAgICAgZGVsZXRlIHRoaXMubm90ZURpY3RbY2hhbm5lbF1bbm90ZV07XG4gICAgICAgICAgICAgICAgICAgIG5vdGVEdXJhdGlvbiA9IHNlc3Npb25UcyAtIG5vdGVEZWYuc2Vzc2lvblRzO1xuICAgICAgICAgICAgICAgICAgICBncmFwaGljcyA9IG5vdGVEZWYuZ3JhcGhpY3M7XG4gICAgICAgICAgICAgICAgICAgIG5vdGVWZWxvY2l0eSA9IG5vdGVEZWYudmVsb2NpdHk7XG4gICAgICAgICAgICAgICAgICAgIHRzID0gbm90ZURlZi50cztcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBub3RlRHVyYXRpb24gPSBEYXRlLm5vdygpIC0gdHM7XG4gICAgICAgICAgICAgICAgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXSA9IHsgdHM6IHRzLCB2ZWxvY2l0eTogdmVsb2NpdHksIHNlc3Npb25Uczogc2Vzc2lvblRzfTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG5cbiAgICAgICAgaWYoIXRoaXMub2Zmc2V0TXVzaWMgfHwgdmVsb2NpdHk9PT0wKSB7XG5cbiAgICAgICAgICAgIHZhciB3aWR0aCA9IG5vdGVEdXJhdGlvbiAqIHRoaXMucGl4ZWxzUGVyU2Vjb25kIC8gMTAwMDtcbiAgICAgICAgICAgIGlmKCFncmFwaGljcykge1xuICAgICAgICAgICAgICAgIHZhciB4ID0gKHRzLXRoaXMuc3RhcnRUcykgKiB0aGlzLnBpeGVsc1BlclNlY29uZCAvIDEwMDA7XG4gICAgICAgICAgICAgICAgaWYoKHgrd2lkdGgpIDwgIChNYXRoLmFicyh0aGlzLmNvbnRhaW5lci54KSAtIHRoaXMud2lkdGgpKSB7XG4gICAgICAgICAgICAgICAgICAgIC8vIG5vdCB2aXNpYmxlLiBkbyBub3RoaW5nXG4gICAgICAgICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgdmFyIHkgPSBNYXRoLmZsb29yKCgxMjgtbm90ZSswLjUpICogdGhpcy5oZWlnaHQgLyAxMjggLSAodGhpcy5ub3RlSGVpZ2h0LzIpKTtcbiAgICAgICAgICAgICAgICB2YXIgY29sb3IgPSB0aGlzLmdldENvbG9yKGNoYW5uZWwpO1xuICAgICAgICAgICAgICAgIHZhciBhbHBoYSA9IChub3RlVmVsb2NpdHkgLyAxMjgpO1xuXG4gICAgICAgICAgICAgICAgZ3JhcGhpY3MgPSB0aGlzLmdldE5vdGVSZWN0KHgsIHksIGNvbG9yLCBhbHBoYSwgd2lkdGgsIHRoaXMubm90ZUhlaWdodCk7XG4gICAgICAgICAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoZ3JhcGhpY3MpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZ3JhcGhpY3Mud2lkdGggPSB3aWR0aDtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgaWYoIWR1cmF0aW9uICYmIHZlbG9jaXR5KSB7XG4gICAgICAgICAgICAgICAgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXS5ncmFwaGljcyA9IGdyYXBoaWNzO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgfTtcblxuICAgIHRoaXMuYWRkTGluZSA9IGZ1bmN0aW9uKHRzKXtcblxuICAgICAgICBpZih0eXBlb2YodHMpID09PSAndW5kZWZpbmVkJykge1xuICAgICAgICAgICAgdHMgPSBuZXcgRGF0ZSgpO1xuICAgICAgICB9XG4gICAgICAgIHZhciB4ID0gLXRoaXMuY29udGFpbmVyLng7XG4gICAgICAgIHZhciB5ID0gdGhpcy5saW5lc0Rvd24gPyB0aGlzLmhlaWdodCAtIDIwIDogMDtcblxuICAgICAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpXG4gICAgICAgICAgICAuYmVnaW5GaWxsKDB4RkZGRjAwKVxuICAgICAgICAgICAgLmxpbmVTdHlsZSgxLCB0aGlzLmxpbmVDb2xvcilcbiAgICAgICAgICAgIC5tb3ZlVG8oMCwgMClcbiAgICAgICAgICAgIC5saW5lVG8oMCwgMjApXG4gICAgICAgICAgICAuZW5kRmlsbCgpO1xuICAgICAgICBncmFwaGljcy54ID0geDtcbiAgICAgICAgZ3JhcGhpY3MueSA9IHk7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGdyYXBoaWNzKTtcbiAgICAgICAgLy8gQWRkIHRleHRcbiAgICAgICAgLy92YXIgdG90YWxTZWMgPSBsaW5lTmIgKiB0aGlzLmxpbmVJbnRlcnZhbCAvIDEwMDA7XG4gICAgICAgIHZhciBob3VycyA9IHRzLmdldEhvdXJzKCk7XG4gICAgICAgIHZhciBtaW51dGVzID10cy5nZXRNaW51dGVzKCk7XG4gICAgICAgIHZhciBzZWNvbmRzID0gdHMuZ2V0U2Vjb25kcygpO1xuICAgICAgICB2YXIgdGltZVN0ciA9IChob3VycyA8IDEwID8gJzAnICsgaG91cnMgOiBob3VycykgKyAnOicgKyAobWludXRlcyA8IDEwID8gJzAnICsgbWludXRlcyA6IG1pbnV0ZXMpICsgJzonICsgKHNlY29uZHMgIDwgMTAgPyAnMCcgKyBzZWNvbmRzIDogc2Vjb25kcyk7XG5cbiAgICAgICAgdmFyIGZvbnRPYmogPSB7IGZvbnQ6ICcxMHB0IEFyaWFsJywgZmlsbDogJyM0NDQ0NDQnIH07XG4gICAgICAgIHZhciB0ID0gbmV3IFBJWEkuVGV4dCh0aW1lU3RyLCBmb250T2JqKTtcbiAgICAgICAgaWYoaXNIb3Jpem9udGFsKSB7XG4gICAgICAgICAgICB0LnggPSB4ICsgMjtcbiAgICAgICAgICAgIHQueSA9IHRoaXMubGluZXNEb3duID8gdGhpcy5oZWlnaHQgLSAxNSA6IDI7XG4gICAgICAgIH1cbiAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICB0LnJvdGF0aW9uID0gLU1hdGguUEkvMjtcbiAgICAgICAgICAgIHQueCA9IHggO1xuICAgICAgICAgICAgdC55ID0gdGhpcy5saW5lc0Rvd24gPyB0aGlzLmhlaWdodCAtIDIgOiB0LndpZHRoICsgMjtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZCh0KTtcbiAgICB9O1xuXG4gICAgdGhpcy5tb3ZlVG8gPSBmdW5jdGlvbihkaWZmVGltZSl7XG4gICAgICAgIHZhciBvbGRYID0gdGhpcy5jb250YWluZXIueDtcbiAgICAgICAgdGhpcy5jb250YWluZXIueCA9IE1hdGguZmxvb3IoZGlmZlRpbWUqdGhpcy5waXhlbHNQZXJTZWNvbmQpO1xuICAgICAgICB2YXIgZGVsdGFYID0gTWF0aC5hYnMob2xkWC10aGlzLmNvbnRhaW5lci54KTtcbiAgICAgICAgXy5mb3JPd24odGhpcy5ub3RlRGljdCwgZnVuY3Rpb24oY2hhbm5lbERpY3QpIHtcbiAgICAgICAgICAgIF8uZm9yT3duKGNoYW5uZWxEaWN0LCBmdW5jdGlvbihub3RlRGVmKSB7XG4gICAgICAgICAgICAgICAgaWYobm90ZURlZi5ncmFwaGljcykge1xuICAgICAgICAgICAgICAgICAgICBub3RlRGVmLmdyYXBoaWNzLndpZHRoID0gbm90ZURlZi5ncmFwaGljcy53aWR0aCArIGRlbHRhWDtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIHRoaXMubW92ZSA9IGZ1bmN0aW9uKCkge1xuICAgICAgICB2YXIgZGlmZiA9ICh0aGlzLnN0YXJ0VHMgLSBEYXRlLm5vdygpKS8xMDAwO1xuICAgICAgICB0aGlzLm1vdmVUbyhkaWZmKTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZW1vdmVQYXNzZWRPYmpldHMgPSBmdW5jdGlvbigpe1xuICAgICAgICB2YXIgY2hpbGRyZW5Ub1JlbW92ZSA9IFtdO1xuICAgICAgICBfKF90aGlzLmNvbnRhaW5lci5jaGlsZHJlbikuZm9yRWFjaChmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICAgICAgcmV0dXJuIHR5cGVvZihjaGlsZCkgPT09ICd1bmRlZmluZWQnIHx8XG4gICAgICAgICAgICAgICAgKGlzSGlkZGVuKGNoaWxkKSAmJiBjaGlsZHJlblRvUmVtb3ZlLnB1c2goY2hpbGQpKTtcbiAgICAgICAgfSk7XG4gICAgICAgIGNoaWxkcmVuVG9SZW1vdmUuZm9yRWFjaChmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICAgICAgX3RoaXMuY29udGFpbmVyLnJlbW92ZUNoaWxkKGNoaWxkKTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgaWYoIXN0YXJ0ZWQpIHtcbiAgICAgICAgICAgIHRoaXMuc3RhcnRUcyA9IERhdGUubm93KCk7XG4gICAgICAgICAgICB0aGlzLmFkZExpbmUoKTtcbiAgICAgICAgICAgIHN0YXJ0ZWQgPSB0cnVlO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMudmVydGljYWxMaW5lc0ludGVydmFsID0gc2V0SW50ZXJ2YWwoZnVuY3Rpb24oKSB7IF90aGlzLmFkZExpbmUoKTsgfSwgdGhpcy5saW5lSW50ZXJ2YWwpO1xuICAgICAgICB0aGlzLmNsZWFuSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbiAoKSB7IF90aGlzLnJlbW92ZVBhc3NlZE9iamV0cygpOyB9LCAxMDAwICogdGhpcy53aWR0aCAvIHRoaXMucGl4ZWxzUGVyU2Vjb25kICk7XG4gICAgfTtcblxuICAgIHRoaXMuc3RvcCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICAvL3dpbmRvdy5jbGVhckludGVydmFsKHRoaXMubW92ZUludGVydmFsKTtcbiAgICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLnZlcnRpY2FsTGluZXNJbnRlcnZhbCk7XG4gICAgICAgIGNsZWFySW50ZXJ2YWwodGhpcy5jbGVhbkludGVydmFsKTtcbiAgICB9O1xuXG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSBQaWFub1JvbGw7XG4iLCIvKipcbiogc2NyaXB0cy9zdGFnZXZpZXcuanNcbipcbiogVGhpcyBpcyB0aGUgc3RhcnRpbmcgcG9pbnQgZm9yIHlvdXIgYXBwbGljYXRpb24uXG4qIFRha2UgYSBsb29rIGF0IGh0dHA6Ly9icm93c2VyaWZ5Lm9yZy8gZm9yIG1vcmUgaW5mb1xuKi9cblxuLyogZ2xvYmFsIGRvY3VtZW50OiBmYWxzZSAqL1xuXG4ndXNlIHN0cmljdCc7XG5cblxudmFyIFBJWEkgPSByZXF1aXJlKCdwaXhpJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xuXG52YXIgZGVmYXVsdENvbmZpZyA9IHtcbiAgICBleHRlcm5hbFJlZnJlc2g6IGZhbHNlLFxuICAgIGxvZ2dlcjogdW5kZWZpbmVkLFxuICAgIHNjZW5lV2lkdGg6IDEwMjQsXG4gICAgc2NlbmVIZWlnaHQ6IDc2OCxcbiAgICBmcmFtZXJhdGU6IDI1LFxuICAgIHNjZW5lQmdDb2xvcjogMHhGRkZGRkYsXG4gICAgY2FudmFzQ29udGFpbmVyOiAnY2FudmFzQ29udGFpbmVyJyxcbn07XG5cbmZ1bmN0aW9uIFN0YWdlVmlldyhvcHRpb25zKSB7XG5cbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0Q29uZmlnKS52YWx1ZSgpO1xuXG4gICAgdmFyIGV4dGVybmFsUmVmcmVzaCA9IG9wdHMuZXh0ZXJuYWxSZWZyZXNoO1xuXG4gICAgdGhpcy5sb2dnZXIgPSBvcHRzLmxvZ2dlcjtcbiAgICB0aGlzLmZyYW1lcmF0ZSA9IG9wdHMuZnJhbWVyYXRlO1xuICAgIHZhciBzY2VuZUJnQ29sb3IgPSBvcHRzLnNjZW5lQmdDb2xvcjtcbiAgICB2YXIgc2NlbmVXaWR0aCA9IG9wdHMuc2NlbmVXaWR0aDtcbiAgICB2YXIgc2NlbmVIZWlnaHQgPSBvcHRzLnNjZW5lSGVpZ2h0O1xuICAgIHZhciBjYW52YXNDb250YWluZXIgPSBvcHRzLmNhbnZhc0NvbnRhaW5lcjtcbiAgICB2YXIgdGltZUNvbnRhaW5lciA9IFtdO1xuICAgIHZhciBjb21wb25lbnRzID0gW107IFxuICAgIFxuICAgIC8vY3JlYXRlIGFuIG5ldyBpbnN0YW5jZSBvZiBhIHBpeGkgc3RhZ2VcbiAgICB0aGlzLnN0YWdlID0gbmV3IFBJWEkuU3RhZ2Uoc2NlbmVCZ0NvbG9yKTtcbiAgICAvL2NyZWF0ZSBhIHJlbmRlcmVyIGluc3RhbmNlLlxuICAgIHZhciByZW5kZXJlciA9IFBJWEkuYXV0b0RldGVjdFJlbmRlcmVyKHNjZW5lV2lkdGgsIHNjZW5lSGVpZ2h0KTtcbiAgICBcdFxuICAgIHRoaXMuaW5pdCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgICAgIGlmKHR5cGVvZihjYW52YXNDb250YWluZXIpID09PSAnc3RyaW5nJykge1xuICAgICAgICAgICAgY2FudmFzQ29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoY2FudmFzQ29udGFpbmVyKTtcbiAgICAgICAgfVxuICAgICAgICBpZih0eXBlb2YodGltZUNvbnRhaW5lcikgPT09ICdzdHJpbmcnKSB7XG4gICAgICAgICAgICB0aW1lQ29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQodGltZUNvbnRhaW5lcik7XG4gICAgICAgIH1cblxuICAgICAgICBjYW52YXNDb250YWluZXIuYXBwZW5kQ2hpbGQocmVuZGVyZXIudmlldyk7XG4gICAgICAgIFxuICAgICAgICBjb21wb25lbnRzLmZvckVhY2goZnVuY3Rpb24oYyl7XG4gICAgXHRcdGMuaW5pdCgpO1xuICAgIFx0fSk7XG4gICAgfTtcbiAgICBcbiAgICB0aGlzLnJlZ2lzdGVyVGltZUNvbnRhaW5lciA9IGZ1bmN0aW9uKGNvbnRhaW5lcikge1xuICAgIFx0dGltZUNvbnRhaW5lci5wdXNoKGNvbnRhaW5lcik7XG4gICAgfTtcbiAgICBcbiAgICB0aGlzLnJlZ2lzdGVyQ29tcG9uZW50ID0gZnVuY3Rpb24oY29tcG9uZW50KSB7XG4gICAgXHRjb21wb25lbnRzLnB1c2goY29tcG9uZW50KTtcbiAgICBcdHRoaXMuc3RhZ2UuYWRkQ2hpbGQoY29tcG9uZW50LmNvbnRhaW5lcik7XG4gICAgfTtcblxuICAgIHRoaXMucmVmcmVzaCA9IGZ1bmN0aW9uKCkge1xuICAgIFx0Y29tcG9uZW50cy5mb3JFYWNoKGZ1bmN0aW9uKGMpe1xuICAgIFx0XHRjLnJlZnJlc2goKTtcbiAgICBcdH0pO1xuICAgICAgICByZW5kZXJlci5yZW5kZXIodGhpcy5zdGFnZSk7XG4gICAgfTtcblxuICAgIC8vIEluaXQgcGFnZSBhbmQgaW50ZXJ2YWxzXG4gICAgdmFyIHJlZnJlc2hJbnRlcnZhbDtcblxuICAgIHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcblxuICAgICAgICBpZighZXh0ZXJuYWxSZWZyZXNoKSB7XG4gICAgICAgICAgICByZWZyZXNoSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbigpIHtfdGhpcy5yZWZyZXNoKCk7fSwgMTAwMC90aGlzLmZyYW1lcmF0ZSk7XG4gICAgICAgIH1cbiAgICAgICAgXG4gICAgICAgIGNvbXBvbmVudHMuZm9yRWFjaChmdW5jdGlvbihjKXtcbiAgICBcdFx0Yy5zdGFydCgpO1xuICAgIFx0fSk7XG4gICAgfTtcblxuICAgIHRoaXMuc3RvcCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBpZighZXh0ZXJuYWxSZWZyZXNoKSB7XG4gICAgICAgICAgICBjbGVhckludGVydmFsKHJlZnJlc2hJbnRlcnZhbCk7XG4gICAgICAgIH1cbiAgICAgICAgY2xlYXJJbnRlcnZhbChyZWZyZXNoVGltZUludGVydmFsKTtcbiAgICAgICAgXG4gICAgICAgIGNvbXBvbmVudHMuZm9yRWFjaChmdW5jdGlvbihjKXtcbiAgICBcdFx0Yy5zdG9wKCk7XG4gICAgXHR9KTtcbiAgICB9O1xuXG5cbiAgICB0aGlzLmxvZyA9IGZ1bmN0aW9uKG0pIHtcbiAgICAgICAgaWYodGhpcy5sb2dnZXIpIHtcbiAgICAgICAgICAgIHRoaXMubG9nZ2VyLmxvZyhtKTtcbiAgICAgICAgfVxuICAgIH07XG5cblxuICAgIHJldHVybiB0aGlzO1xufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBTdGFnZVZpZXc6IFN0YWdlVmlld1xufTtcbiIsIi8qKlxuKiBqcy91dGlscy5qc1xuKlxuKiBiYXNpYyB0b29sc1xuKlxuKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG5mdW5jdGlvbiBmb3JtYXRUaW1lICh0cykge1xuXHR2YXIgaG91cnMgPSBNYXRoLmZsb29yKCAodHMvMTAwMCkgLyAzNjAwICkgJSAyNDtcblx0dmFyIG1pbnV0ZXMgPSBNYXRoLmZsb29yKCAodHMvMTAwMCkgLyA2MCApICUgNjA7XG5cdHZhciBzZWNvbmRzID0gTWF0aC5mbG9vciggKHRzLzEwMDApICUgNjApO1xuXHRyZXR1cm4gKChob3VycyA8IDEwID8gJzAnICsgaG91cnMgOiBob3VycykgKyAnOicgKyAobWludXRlcyA8IDEwID8gJzAnICsgbWludXRlcyA6IG1pbnV0ZXMpICsgJzonICsgKHNlY29uZHMgIDwgMTAgPyAnMCcgKyBzZWNvbmRzIDogc2Vjb25kcykpO1xufVxuXG5cbm1vZHVsZS5leHBvcnRzID0ge1xuXHRmb3JtYXRUaW1lOiBmb3JtYXRUaW1lXG59O1xuIiwiLyoqXG4qIGpzL3dzd3JhcHBlci5qc1xuKlxuKiBzaW1wbGUgd2Vic2VydmljZSB3cmFwcGVyIHRvIHJlZ2lzdGVyIGNhbGxiYWNrcyBvbiBvbm1lc3NhZ2VcbipcbiovXG5cbi8qIGdsb2JhbCBXZWJTb2NrZXQ6IGZhbHNlICovXG5cbid1c2Ugc3RyaWN0JztcblxuZnVuY3Rpb24gV3NXcmFwcGVyKHdzdXJsLCBsb2dnZXIpIHtcblxuICAgIHZhciB1cmwgPSB3c3VybDtcbiAgICB2YXIgc29jayA9IG5ldyBXZWJTb2NrZXQodXJsKTtcbiAgICB2YXIgbG9nZ2VyT2JqID0gbG9nZ2VyO1xuXG4gICAgdmFyIGxvZyA9IGZ1bmN0aW9uKG1zZykge1xuICAgICAgICBpZihsb2dnZXJPYmopIHtcbiAgICAgICAgICAgIGxvZ2dlck9iai5sb2cobXNnKTtcbiAgICAgICAgfVxuICAgIH07XG5cbiAgICB2YXIgaGFuZGxlcnMgPSBbXTtcblxuICAgIHNvY2sub25vcGVuID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIGxvZygnQ29ubmVjdGVkIHRvICcgKyB1cmwpO1xuICAgIH07XG5cbiAgICBzb2NrLm9uY2xvc2UgPSBmdW5jdGlvbihlKSB7XG4gICAgICAgIGxvZygnQ29ubmVjdGlvbiBjbG9zZWQgKHdhc0NsZWFuID0gJyArIGUud2FzQ2xlYW4gKyAnLCBjb2RlID0gJyArIGUuY29kZSArICcsIHJlYXNvbiA9IFxcJycgKyBlLnJlYXNvbiArICdcXCcpJyk7XG4gICAgICAgIHNvY2sgPSBudWxsO1xuICAgIH07XG5cbiAgICBzb2NrLm9ubWVzc2FnZSA9IGZ1bmN0aW9uKGUpIHtcbiAgICAgICAgbG9nKCdyZWNlaXZlZCAnICsgZS5kYXRhKTtcbiAgICAgICAgdmFyIGRhdGEgPSBKU09OLnBhcnNlKGUuZGF0YSk7XG4gICAgICAgIGhhbmRsZXJzLmZvckVhY2goZnVuY3Rpb24oaGFuZGxlcikge1xuICAgICAgICAgICAgaGFuZGxlcihkYXRhKTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIHRoaXMubWVzc2FnZSA9IGZ1bmN0aW9uKGhhbmRsZXIpIHtcbiAgICAgICAgaWYoaGFuZGxlcikge1xuICAgICAgICAgICAgaGFuZGxlcnMucHVzaChoYW5kbGVyKTtcbiAgICAgICAgfVxuICAgIH07XG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgV3NXcmFwcGVyOiBXc1dyYXBwZXJcbn07XG4iXX0=