annot-server/static/js/annotviz.js
author ymh <ymh.work@gmail.com>
Fri, 17 Apr 2015 15:32:19 +0200
changeset 166 fa9e24e46968
parent 161 0b01cbd8ed9e
permissions -rw-r--r--
pianoroll use ts_server

!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){
/**
 * scripts/main.js
 *
 * This is the starting point for your application.
 * Take a look at http://browserify.org/ for more info
 */

'use strict';

var doubleroll = require('./doubleroll');
var annotsroll = require('./annotsroll');
var annotstimeline = require('./annotstimeline');
var annotsvizview = require('./annotsvizview');
var stageview = require('./stageview');
var wswrapper = require('./wswrapper');
var logger = require('./logger');
var utils = require('./utils');

var _ = require('lodash');

module.exports = _({})
    .extend(doubleroll)
    .extend(annotsroll)
    .extend(annotstimeline)
    .extend(annotsvizview)
    .extend(stageview)
    .extend(wswrapper)
    .extend(logger)
    .extend(utils)
    .value();

},{"./annotsroll":2,"./annotstimeline":3,"./annotsvizview":4,"./doubleroll":5,"./logger":6,"./stageview":8,"./utils":9,"./wswrapper":10,"lodash":"lodash"}],2:[function(require,module,exports){
/**
* js/annotsRoll.js
*
* annotsRoll basic component
*
*/

'use strict';

var PIXI = require('pixi');
var _ = require('lodash');

var DEFAULT_ANNOT_COLOR = '#bababa';

var defaultAnnotStyles = {
    'label': { font: '16pt Arial Bold', fill: '#65A954', wordWrap: true},
    'text' : { font: '12pt Arial Regular', fill: '#444444', wordWrap: true},
    'user' : { font: '14pt Arial regular', fill: '#666666' },
};

var defaultOptions = {
    externalRefresh: false,
    defaultColor: DEFAULT_ANNOT_COLOR,
    annotStyles: defaultAnnotStyles,
    ignoreAnnots:false
};

function AnnotsRoll(options) {

//parentContainer, xInit, yInit, width, height, widthRoll, pixelsPerSecond, annotColors
    var _this = this;
    var opts = _(options).defaults(defaultOptions).value();


    this.container = new PIXI.DisplayObjectContainer();
    this.container.x = opts.xInit;
    this.container.y = opts.yInit;
    this.container.width = opts.width;

    this.height = opts.height;
    this.width = opts.width;
    this.widthRoll = opts.widthRoll;
    this.pixelsPerSecond = opts.pixelsPerSecond;
    this.annotColors = opts.annotColors;
    this.startTs = opts.startTs || Date.now();
    this.ignoreAnnots = opts.ignoreAnnots;

    var yInit = opts.yInit;
    var annotStyles = _(opts.annotStyles).defaults(defaultAnnotStyles).value();
    var marginX = 15;
    for(var style in annotStyles) {
    	if (annotStyles[style].wordWrap === true){
    		annotStyles[style].wordWrapWidth = this.widthRoll - marginX;
    	}
    }
    var started = false;
    var ws = opts.ws;
    var externalRefresh = opts.externalRefresh;
    var stageView = opts.stageView;
    var waitInterval;
    var wait = 0;

    stageView.registerComponent(this);

    var isHidden = function(child) {
        // TODO: the origin point is an approximation. Should refine this
        var globalPos = child.toGlobal(new PIXI.Point(0,0));
        return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ;
    };

    this.addAnnots = function(data) {

        //var title = data.content.category.label;
        //var user = data.content.user;
        //Test cat and color
        //var colorAnnot = 0x65A954;
        var category = data.content.category.label,
            text     = data.content.text,
            user     = data.content.user,
            ts       = Date.parse(data.ts),
            color    = data.content.color || this.getColor(ts, data.content.category.code);

        this.addAnnot(category, text, user, color, ts);
    };

    this.getColor = function(ts, code) {
        var colorsDef;
        _(this.annotColors).eachRight(function(cdef) {
            if(cdef.ts < ts) {
                colorsDef = cdef.colors;
                return false;
            }
        });
        var resColor;
        if(colorsDef) {
            resColor = colorsDef[code];
        }
        if(!resColor) {
            resColor = colorsDef.defaultColor || DEFAULT_ANNOT_COLOR;
        }
        return resColor;
    };

    this.addAnnot = function(category, text, user, color, ts){

        var x = 0;
        var y = (ts-this.startTs) * this.pixelsPerSecond / 1000 + yInit;

        var colorHex = parseInt(color.replace(/^#/, ''), 16);

        if (wait === 0){
	        var graphics = new PIXI.Graphics()
	            .beginFill(colorHex)
	            .drawRect(x, y, 10, 3)
	            .endFill();

	        this.container.addChild(graphics);

	        var textHeight = 0;
	        var catLabel = new PIXI.Text(
	            category,
	            _(annotStyles.label).extend({fill: color}).value()
	        );
	        catLabel.x = x + marginX;
	        catLabel.y = y - 23;
	        this.container.addChild(catLabel);
	        textHeight += (catLabel.height - 23 + 2);

	        if(text) {
	            var catText = new PIXI.Text(text, annotStyles.text);
	            catText.x = x + marginX;
	            catText.y = y + textHeight;
	            this.container.addChild(catText);
	            textHeight += (catText.height + 2);
	        }

	        var catUser = new PIXI.Text(user, annotStyles.user);
	        catUser.x = x + marginX;
	        catUser.y = y + textHeight;
	        this.container.addChild(catUser);
	        textHeight += (catUser.height + 8);

	        if (this.ignoreAnnots === true){
		        wait = textHeight / this.pixelsPerSecond;
		        waitInterval = setInterval(function() {_this.refreshWait();}, 1000);
	        }
        }

        this.addAnnotLine(colorHex, y);
    };

    this.addAnnotLine = function(color, y) {
        var x = this.widthRoll;


        var graphics = new PIXI.Graphics()
            .beginFill(color)
            .drawRect(x, y, this.width - x, 3)
            .endFill();

        this.container.addChild(graphics);
    };

    this.moveTo = function(diffTime){
    	this.container.y = Math.floor(diffTime*this.pixelsPerSecond);
    };

    this.move = this.refresh = function() {
        var diff = (this.startTs - Date.now())/1000;
        this.moveTo(diff);
    };

    this.refreshWait = function(){
    	wait -= 1;
    	if (wait < 0){
    		wait = 0;
    		clearInterval(waitInterval);
    	}
    };

    this.removePassedObjets = function(){
        var childrenToRemove = [];
        _(_this.container.children).forEach(function(child) {
            return typeof(child) === 'undefined' ||
                (isHidden(child) && childrenToRemove.push(child));
        });
        childrenToRemove.forEach(function(child) {
            _this.container.removeChild(child);
        });
    };

    this.init = function() {

        ws.message(function(data) {
            _this.addAnnots(data);
        });

    };


    this.start = function() {
        if(!started) {
            this.startTs = Date.now();
            started = true;
        }
        this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.height / this.pixelsPerSecond );
        if(!externalRefresh) {
            this.refreshInterval = setInterval(function() {_this.move();}, 1000/this.framerate);
        }
    };

    this.stop = function() {
        clearInterval(this.cleanInterval);
        if(!externalRefresh) {
            clearInterval(this.refreshInterval);
        }
    };

}

module.exports = {
    AnnotsRoll: AnnotsRoll,
};

},{"lodash":"lodash","pixi":"pixi"}],3:[function(require,module,exports){
/**
* js/annotstimeline
*
* annotstimeline basic component
*
*/

'use strict';

var PIXI = require('pixi');
var Utils = require('./utils.js');
var _ = require('lodash');

var defaultOptions = {
    logger: undefined,
    intervalWidth: 10,
    intervalHeight: 5,
    maxCellHeight: 200,
    radius: 300,
    serverUrl: 'http://127.0.0.1:8080',
    channel: 'ANNOT',
    maxPages: 1000,
    showClockGraphics: true,
    archive: false
};


function AnnotsTimeLine(options){
    var _this = this;
    var opts = _(options).defaults(defaultOptions).value();

    this.container = new PIXI.DisplayObjectContainer();
    this.container.x = opts.xInit;
    this.container.y = opts.yInit;
    this.container.width = opts.width;
    this.container.height = opts.height;

    this.timeBegin = opts.timeBegin;
    this.timeEnd = opts.timeEnd;
    this.duration = (this.timeEnd - this.timeBegin)/1000;
    this.width = opts.width;
    this.height = opts.height;
    this.intervalHeight = opts.intervalHeight;
    this.intervalWidth = opts.intervalWidth;
    this.maxCellHeight = opts.maxCellHeight;
    this.annotCategories = opts.annotCategories;
    this.startTs = options.startTs || Date.now();
    this.showClockGraphics = opts.showClockGraphics;
    this.archive = opts.archive;

    this.circleX = opts.circleX || (this.width/2);
    this.circleY = opts.circleY || (this.height/2);
    this.radius = opts.radius;
    var perimeter = 2*Math.PI* this.radius;
    this.intervalDuration = (this.intervalWidth * this.duration / perimeter);

    var channel = opts.channel;
    var eventCode = opts.eventCode;
    var serverUrl = opts.serverUrl;
    var maxPages = opts.maxPages;

    var totalIndex = Math.floor( perimeter/this.intervalWidth);

    this.cells = [];
    for (var i=0; i<(perimeter/this.intervalWidth) ; i++){
        this.cells[i] = [];
        this.cells[i].i = i;
        this.cells[i].totalAnnots = 0;
        this.cells[i].categories = {};
    }

    var ws = opts.ws;
    var stageView = opts.stageView;

    //draw the base - circle and line to locate the scene
    var graphics = new PIXI.Graphics();
    graphics.lineStyle(2, 0x646464)
        .drawCircle(this.circleX, this.circleY, this.radius - 3)
        .endFill();
    this.container.addChild(graphics);

    stageView.registerComponent(this);

    var loadArchives = function() {
        //start timeBegin end startTime
        //query -> need channel + eventCode
        //iterate over data fill cells
        var startTs = _this.timeBegin;
        var endTs = Math.min(_this.timeEnd,_this.startTs);

        var url = serverUrl + '/p/api/v1/annotation';
        var filters = [
            { name: 'ts', op: '>', val: new Date(startTs).toISOString()}, //start
            { name: 'ts', op: '<=', val: new Date(endTs).toISOString()}, //end
            { name: 'channel', op: '==', val: channel}, //channel
            { name: 'event_code', op: '==', val: eventCode} //eventcode
        ];

        url = url + '?q=' + JSON.stringify({filters:filters});

        var totalPage = 1;
        var currentPage = 1;

        var processResFunction = function(res) {
            //console.log("RES archive", res);
            if(res) {
                var data = res.target.json;
                /*jshint -W069 */
                totalPage = Math.min(maxPages,parseInt(data['total_pages']));
                data.objects.forEach(function(annotation) {
                    _this.addAnnot(annotation);
                });
            }
            if(currentPage <= totalPage) {
                var jsonLoader = new PIXI.JsonLoader(url+'&page='+currentPage, true);
                jsonLoader.on('loaded', processResFunction);
                jsonLoader.load();
                currentPage++;
            }
        };
        processResFunction();

    };

    //Add Annotation to the TimeLine
    this.addAnnot = function(data){

        var ts = Date.parse(data.ts);
        var colorsDef;
        _(this.annotCategories).eachRight(function(cdef) {
            if(cdef.ts < ts) {
                colorsDef = cdef;
                return false;
            }
        });

        if(!colorsDef) {
            return;
        }

        if (this.timeEnd > ts){
            var i = Math.floor((ts - this.timeBegin)/(1000*this.intervalDuration));

            if (typeof(this.cells[i].graphics) === 'undefined'){
                this.initCell(this.cells[i], colorsDef);
            }

            var annotCode;
            if (typeof(colorsDef.colors[data.content.category.code]) !== 'undefined'){
                annotCode = data.content.category.code;
            } else {
                annotCode = 'default';
            }

            this.cells[i].categories[annotCode].count += 1;
            this.cells[i].totalAnnots +=1;
            this.redrawCell(this.cells[i], colorsDef);
        }
    };

    this.initClockGraphics = function() {
        var tBeg = new PIXI.Text(Utils.formatTime(this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
        tBeg.x = this.circleX + 15;
        tBeg.y = this.circleY - this.radius - this.maxCellHeight - 10;
        this.container.addChild(tBeg);

        var tEnd = new PIXI.Text(Utils.formatTime(this.timeEnd), { font: '12pt Gothic Standard', fill: '#646464' });
        tEnd.x = this.circleX - 15 - tEnd.width;
        tEnd.y = this.circleY - this.radius - this.maxCellHeight - 10;
        this.container.addChild(tEnd);

        var t15 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
        t15.x = this.circleX + this.radius + this.maxCellHeight + 10 ;
        t15.y = this.circleY - t15.height;
        t15.rotation = Math.PI /2;
        this.container.addChild(t15);

        var t30 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/2) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
        t30.x = this.circleX - t30.width/2;
        t30.y = this.circleY + this.radius + this.maxCellHeight - 2;
        this.container.addChild(t30);

        var t45 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)*3/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
        t45.x = this.circleX - this.radius - this.maxCellHeight - 10 ;
        t45.y = this.circleY + t15.height;
        t45.rotation = -Math.PI/2;
        this.container.addChild(t45);

        var lineV = new PIXI.Graphics();
        lineV.lineStyle(1, 0x646464)
            .moveTo(this.circleX, this.circleY - (this.radius/3)/2)
            .lineTo(this.circleX, this.circleY - this.radius - this.maxCellHeight - 10)
            .endFill();
        this.container.addChild(lineV);
    };

    //Draw the cellule
    this.redrawCell = function(cell, colorsDef){
        var y = 0;

        //Check if total height is higher than Max Cell Height
        var heightStep;
        if ((cell.totalAnnots*this.intervalHeight) > this.maxCellHeight){
            heightStep = this.maxCellHeight/cell.totalAnnots;
        } else {
            heightStep = this.intervalHeight;
        }

        //Draw the rect depending on the height step calculated
        for (var i=0; i< colorsDef.order.length; i++){
            var currentCode = colorsDef.order[i];
            cell.graphics.beginFill(cell.categories[currentCode].color.replace('#', '0x'))
                .drawRect(0, y, this.intervalWidth-1, -cell.categories[currentCode].count * heightStep)
                .endFill();
            y -= cell.categories[currentCode].count*heightStep;
        }
    };

    this.initCell = function(cell, colorsDef){
        cell.graphics = new PIXI.Graphics();
        cell.graphics.position.x = this.circleX + this.radius * Math.sin(cell.i*(360/totalIndex)*(Math.PI/180));
        cell.graphics.position.y = this.circleY - this.radius * Math.cos(cell.i*(360/totalIndex)*(Math.PI/180));
        cell.graphics.rotation = (cell.i)*(360/totalIndex)*(Math.PI/180) + (360/(totalIndex*2))*(Math.PI/180);
        this.container.addChild(cell.graphics);

        for (var category in colorsDef.colors){
            cell.categories[category] = {
                'count': 0,
                'color': colorsDef.colors[category]
            };
        }
        if (typeof(cell.categories['default']) === 'undefined'){
            cell.categories['default'] = {
                'count': 0,
                'color': colorsDef.defaultColor
            };
        }
    };

    this.init = function() {
        if (!this.archive){
            ws.message(function(data) {
                _this.addAnnot(data);
            });
        }

        if (this.showClockGraphics){this.initClockGraphics();}
    };


    this.start = function() {
        this.startTs = Date.now();
        loadArchives();
    };

    this.refresh = function() {

    };

    this.stop = function(){
    };

    return this;
}

module.exports = {
    AnnotsTimeLine: AnnotsTimeLine
};

},{"./utils.js":9,"lodash":"lodash","pixi":"pixi"}],4:[function(require,module,exports){
/**
* js/annotsvizview.js
*
* This is the starting point for your application.
* Take a look at http://browserify.org/ for more info
*
*/
/*jshint unused: false */
'use strict';

var PIXI = require('pixi');
var _ = require('lodash');
var DoubleRoll = require('./doubleroll.js');
var AnnotsTimeLine = require('./annotstimeline.js');
var AnnotsRoll = require('./annotsroll.js');
var Utils = require('./utils.js');

var defaultOptions = {
    xInit: 0,
    yInit: 0,
    width: 1024,
    height: 768
};

function AnnotsVizView(options){
    var _this = this;
    var opts = _(options).defaults(defaultOptions).value();

    this.container = new PIXI.DisplayObjectContainer();
    this.container.x = opts.xInit;
    this.container.y = opts.yInit;
    this.width = opts.width;
    this.height= opts.height;
    this.timeBegin = opts.timeBegin;
    this.timeEnd = opts.timeEnd;

    this.annotCategories = [];
    this.annotCategoriesDay2 = [];
    this.annotCategoriesDay1 = [];

    var wsPianoroll = opts.wsPianoroll;
    var wsAnnot = opts.wsAnnot;
    var stageView = opts.stageView;
    var channel = opts.channel;
    var serverUrl = opts.serverUrl;

    var eventCode = opts.eventCode;
    var eventCodeSessionDay2 = opts.eventCodeSessionDay2;
    var eventCodeSessionDay1 = opts.eventCodeSessionDay1;
    var logger = opts.logger;

    Utils.getAnnotCategories(opts.urlCategories + eventCode, this.annotCategories);
    Utils.getAnnotCategories(opts.urlCategories + eventCodeSessionDay2, this.annotCategoriesDay2);
    Utils.getAnnotCategories(opts.urlCategories + eventCodeSessionDay1, this.annotCategoriesDay1);

    stageView.registerComponent(this);

    var currentDay = new AnnotsTimeLine.AnnotsTimeLine({
        stageView : stageView,
        logger: logger,
        ws: wsAnnot,
        xInit: 0,
        yInit: 0,
        width: 1024 - 200 - 200,
        height: 768-200,
        timeBegin: this.timeBegin,
        timeEnd: this.timeEnd,
        intervalWidth: 6,
        intervalHeight: 10,
        maxCellHeight: 70,
        radius: 200,
        eventCode: eventCode,
        channel: channel,
        serverUrl: serverUrl,
        annotCategories: this.annotCategories
    });

    //Archive day 2
    var timeLineDay2 = new AnnotsTimeLine.AnnotsTimeLine({
        stageView : stageView,
        xInit: 0,
        yInit: 0,
        width: 1024 - 200 - 200,
        height: 768-200,
        timeBegin: Date.parse('2015-04-16T14:00:00+02:00'),
        timeEnd: Date.parse('2015-04-16T21:00:00+02:00'),
        circleX: currentDay.circleX,
        circleY: currentDay.circleY,
        intervalWidth: (currentDay.radius*2/3)* currentDay.intervalWidth / currentDay.radius,
        intervalHeight: (currentDay.intervalHeight * (currentDay.radius - currentDay.radius*2/3))/ currentDay.maxCellHeight,
        maxCellHeight: (currentDay.radius - currentDay.radius*2/3)/2,
        radius: currentDay.radius*2/3,
        annotCategories: this.annotCategoriesDay2,
        eventCode: eventCodeSessionDay2,
        channel: channel,
        serverUrl: serverUrl,
        showClockGraphics:false,
        archive: true
    });

    //Archive day 1
    var timeLineDay1 = new AnnotsTimeLine.AnnotsTimeLine({
        stageView : stageView,
        xInit: 0,
        yInit: 0,
        width: 1024 - 200 - 200,
        height: 768-200,
        timeBegin: Date.parse('2015-04-16T14:00:00+02:00'),
        timeEnd: Date.parse('2015-04-16T21:00:00+02:00'),
        circleX: currentDay.circleX,
        circleY: currentDay.circleY,
        intervalWidth: (currentDay.radius*2/3)* currentDay.intervalWidth / currentDay.radius,
        intervalHeight: (currentDay.intervalHeight * (currentDay.radius - currentDay.radius/3))/ currentDay.maxCellHeight,
        maxCellHeight: (currentDay.radius*2/3 - currentDay.radius/3)/4,
        radius: currentDay.radius/3,
        annotCategories: this.annotCategoriesDay1,
        eventCode: eventCodeSessionDay1,
        channel: channel,
        serverUrl: serverUrl,
        showClockGraphics:false,
        archive: true,
    });

    var currentTimeText = new PIXI.Text('-- : -- : --', { font: '18pt Gothic Standard', fill: '#646464' });
    currentTimeText.x = timeLineDay2.circleX - currentTimeText.width/2;
    currentTimeText.y = timeLineDay2.circleY - currentTimeText.height/2;
    this.container.addChild(currentTimeText);


    var doubleRollH = new DoubleRoll.DoubleRoll({
        stageView : stageView,
        logger: logger,
        ws: wsPianoroll,
        yInit: (this.height - 200),
        sceneHeight: 200,
        pianorolls : [
            {
                height: 200,
                timeWidth: 10,
                lineInterval: 5000,
                noteHeight: 10
            },
        ]
    });

    var doubleRollV = new DoubleRoll.DoubleRoll({
        stageView : stageView,
        logger: logger,
        ws: wsPianoroll,
        orientation: 'vertical',
        sceneHeight: 768-200,
        pianorolls : [
            {
                height: 200,
                timeWidth: 60,
                lineInterval: 5000,
                noteHeight: 5,
            },
        ]
    });

    var annotsRoll = new AnnotsRoll.AnnotsRoll({
        stageView : stageView,
        logger: logger,
        ws: wsAnnot,
        parentContainer: doubleRollV.stage,
        xInit: 1024 - 200 - 200,
        yInit: 768-200,
        width: 200 + 200,
        height: 768-200,
        widthRoll: 200,
        framerate: doubleRollV.framerate,
        pixelsPerSecond: Math.floor(1024 / 60),
        annotColors: this.annotCategories
    });

    var limiters = new PIXI.Graphics()
        .lineStyle(1, 0x646464)
        .moveTo(annotsRoll.container.x, annotsRoll.container.y)
        .lineTo(annotsRoll.container.x, annotsRoll.container.y - annotsRoll.height)
        .moveTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y)
        .lineTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y - annotsRoll.height)
        .moveTo(0, this.height - 200)
        .lineTo(this.width, this.height - 200)
        .drawRect(0, 0, this.width -1, this.height -1)
        .beginFill(0xECECEC)
        .drawRect(1024 - 200, 0, 200, 768-200)
        .endFill();
    this.container.addChild(limiters);

    this.init = function(){
    };

    this.updateTime = function(){
        currentTimeText.setText(Utils.formatTime(Date.now()));
    };

    var refreshTimeInterval;

    this.start = function() {
        refreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000);
    };

    this.refresh = function() {
    };

    this.stop = function(){
    };

    return this;

}

module.exports = {
    AnnotsVizView: AnnotsVizView
};

},{"./annotsroll.js":2,"./annotstimeline.js":3,"./doubleroll.js":5,"./utils.js":9,"lodash":"lodash","pixi":"pixi"}],5:[function(require,module,exports){
/**
* scripts/doubleroll.js
*
* This is the starting point for your application.
* Take a look at http://browserify.org/ for more info
*/

'use strict';


var PIXI = require('pixi');
var _ = require('lodash');
var PianoRoll = require('./pianoroll');
var Utils = require('./utils.js');

var defaultConfig = {
    orientation: 'horizontal',
    logger: undefined,
    sceneWidth: 1024,
    showTimer: false,
    pianorolls : [
      {
        height: 384,
        timeWidth: 10,
        lineInterval: 5000,
        noteHeight: undefined,
        range: {
    		bottom: 40,
    		top: 90,
        },
        dynamicRange: true,
      },
      {
        height: 384,
        timeWidth: 60,
        lineInterval: 5000,
        noteHeight: undefined,
        range:{
    		bottom: 0,
    		top: 128,
        },
        dynamicRange: false,
      },
    ],
    framerate: 25,
    offsetMusic: false,
    sceneBgColor: 0xFFFFFF,
    lineColor: 0x444444,
    lineFillColor: 0xFFFF00,
    noteColors: [0xB90000, 0x4BDD71, 0xAF931E, 0x1C28BA, 0x536991],
    noteHeight: undefined,
    zeroShift: 0.9,
    timeWidth: 60,
    lineInterval: 5000,
//    wsUri: undefined,
//    eventCode: undefined

};

function DoubleRoll(options) {

    var _this = this;
    var opts = _(options).defaults(defaultConfig).value();

    var orientation = opts.orientation;
    var isHorizontal = (orientation !== 'vertical');

    this.logger = opts.logger;
    this.lineColor = opts.lineColor;
    this.lineFillColor = opts.lineFillColor;
    this.framerate = opts.framerate;
    this.offsetMusic = opts.offsetMusic;
    this.noteColors = opts.noteColors;
    this.showTimer = opts.showTimer;

    var noteHeight = opts.noteHeight;
    var sceneHeight = opts.sceneHeight || _(opts.pianorolls).reduce(function(s,p) { return s + p.height; }, 0);
    var timeWidth = opts.timeWidth;
    var lineInterval = opts.lineInterval;
    var offsetMusic = opts.offsetMusic;

    var sceneWidth = opts.sceneWidth;
    var stageView = opts.stageView;

    
    var zeroShift = opts.zeroShift;

    var ws = opts.ws;

    var colorsReg = {};

    this.container = new PIXI.DisplayObjectContainer();
    this.container.x = Math.floor(sceneWidth*zeroShift);
    this.container.y = 0;

    stageView.registerComponent(this);

    var pianorollList = [];

    var pianorollOptions = {
        parentContainer: this.container,
        orientation: orientation,
        xInit: 0,
        width: sceneWidth,
        noteColors: this.noteColors,
        colorsReg: colorsReg,
        lineColor: this.lineColor,
        lineInterval: lineInterval,
        offsetMusic: offsetMusic,
    };

    var yInit = opts.yInit || 0;
    var linesDown = true;
    _(opts.pianorolls).forEach(function(prDef, i) {
        var prNoteHeight = noteHeight || prDef.noteHeight || prDef.height / (prDef.range.top - prDef.range.bottom + 1);
        var prTimeWidth = prDef.timeWidth || timeWidth;
        pianorollList.push(new PianoRoll(_({
            yInit: yInit,
            height: prDef.height,
            linesDown: linesDown,
            pixelsPerSecond: Math.floor(sceneWidth / prTimeWidth),
            noteHeight: prNoteHeight,
            lineInterval: prDef.lineInterval,
            range: prDef.range,
            dynamicRange: prDef.dynamicRange,
        }).defaults(pianorollOptions).value()));
        yInit += prDef.height;
        linesDown = !linesDown;

        if(i<(opts.pianorolls.length-1)) {
            var lineGraphics = new PIXI.Graphics()
                .beginFill(_this.lineFillColor)
                .lineStyle(1, _this.lineColor)
                .moveTo(Math.floor(sceneWidth*zeroShift), yInit)
                .lineTo(-sceneWidth - Math.floor(sceneWidth*zeroShift), yInit)
                .endFill();
            _this.container.addChild(lineGraphics);
        }
    });

    if (this.showTimer){
	    var currentTimeText = new PIXI.Text('-- : -- : --', { font: '40pt Arial', fill: '#646464' });
	    currentTimeText.x = -currentTimeText.width - 60; 
	    currentTimeText.y = sceneHeight - currentTimeText.height - 30;
	    this.container.addChild(currentTimeText);
	}
    
    
    if(!isHorizontal) {
        this.container.rotation = Math.PI/2; 
        this.container.y = sceneHeight;
        this.container.x = sceneWidth;
    }


    this.init = function() {

    	ws.message(function(data) {
            _this.addNotes(data);
        });

    };
    
    this.updateTime = function(){
        currentTimeText.setText(Utils.formatTime(Date.now()));
    };

    this.addNotes = function(data) {

        pianorollList.forEach(function(c) {
            c.addNoteRaw(data);
        });
    };

    this.refresh = function() {
        pianorollList.forEach(function(c) {
            c.move();
        });
    };

    // Init page and intervals
    var startTs;
    var refreshTimeInterval;

    this.start = function() {
    	if (this.showTimer){
    		refreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000);
    	}
    	
        startTs = Date.now();
        pianorollList.forEach(function(c) {
            c.start();
        });
    };

    this.stop = function() {

        pianorollList.forEach(function(c) {
            c.stop();
        });
    };


    this.log = function(m) {
        if(this.logger) {
            this.logger.log(m);
        }
    };



    return this;
}

module.exports = {
    DoubleRoll: DoubleRoll
};

},{"./pianoroll":7,"./utils.js":9,"lodash":"lodash","pixi":"pixi"}],6:[function(require,module,exports){
/**
* js/wswrapper.js
*
* simple logger service
*
*/

/* global document: false */

'use strict';

function HtmlLogger(doLog, container) {

    var logContainer = container;
    if(typeof(container) === 'string') {
        logContainer = document.getElementById(container);
    }
    if(!doLog) {
        document.body.removeChild(logContainer);
        logContainer = undefined;
    }


    this.log = function(msg) {
        if(doLog && logContainer) {
            logContainer.innerHTML += msg + '\n';
            logContainer.scrollTop = logContainer.scrollHeight;
        }
    };
}

function ConsoleLogger(doLog) {

    this.log = function(msg) {
        if(doLog) {
            console.log(msg);
        }
    };

}

module.exports = {
    HtmlLogger: HtmlLogger,
    ConsoleLogger: ConsoleLogger
};

},{}],7:[function(require,module,exports){
/**
* js/pianoroll.js
*
* pianoroll basic component
*
*/

//TODO: add delay to adjust the "zero". this could should be given as an option
//or if explicitely null automatically adjusted from the mean of difference
//between the note timestamp and the current timestamp.

'use strict';


var PIXI = require('pixi');
var randomColor = require('randomColor');
var _ = require('lodash');

var NTP_EPOCH_DELTA = 2208988800; //c.f. RFC 868

function PianoRoll(options) {
    var _this = this;
    this.container = new PIXI.DisplayObjectContainer();
    this.container.x = options.xInit;
    this.container.y = options.yInit;
    options.parentContainer.addChild(this.container);

    var orientation = options.orientation;
    var isHorizontal = (orientation !== 'vertical');

    this.linesDown = options.linesDown;
    this.height = options.height;
    this.pixelsPerSecond = options.pixelsPerSecond;
    this.width = options.width;
    this.noteColors = options.noteColors;
    this.colorsReg = options.colorsReg || {};
    this.lineColor = options.lineColor;
    this.lineInterval = options.lineInterval;
    this.offsetMusic = options.offsetMusic || false;
    this.noteHeight = options.noteHeight;
    this.noteDict = {};
    this.startTs = options.startTs || Date.now();
    this.dynamicRange = options.dynamicRange;
    this.initRange = options.range;
    this.range = options.range;


    var started = false;

    var isHidden = function(child) {
        // TODO: the origin point is an approximation. Should refine this
        var globalPos = child.toGlobal(new PIXI.Point(0,0));
        return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ;
    };

    //TODO: I do not like the "regColor" object. This should not be global, but local
    this.getColor = function(canal) {
        var color = this.colorsReg[canal];
        if(typeof(color) === 'undefined') {
            var colorsRegSize = Object.keys(this.colorsReg).length;
            if(colorsRegSize < this.noteColors.length) {
                color = this.colorsReg[canal] = this.noteColors[colorsRegSize];
            }
            else {
                color = this.colorsReg[canal] = parseInt(randomColor({ luminosity: 'light', hue: 'random', format:'hex'}).replace(/^#/, ''), 16);
            }
        }
        return color;
    };

    this.getNoteRect = function(note, x, y, color, alpha, width, height) {
        var graphics = new PIXI.Graphics();
        graphics.note = note;
        graphics.beginFill(color, alpha);
        graphics.drawRect(0, 0, width, height);
        graphics.endFill();
        graphics.x = x;
        graphics.y = y;
        graphics.width = width;
        graphics.height = height;
        return graphics;
    };

    this.addNoteRaw = function(data) {
        var note = data.content[3];
        var velocity = data.content[4];
        //var ts = (data.content[0] - NTP_EPOCH_DELTA)*1000;
        var ts = Date.parse(data.ts_server);
        console.log(ts, data.content[0]);
        var channel = data.content[2];
        var sessionTs = data.content[1];

        if(velocity !== 0 && typeof(this.noteDict[channel]) !== 'undefined' && typeof(this.noteDict[channel][note] !== 'undefined')) {
            this.addNote(note, ts, sessionTs, 0, channel, 0);
        }

        this.addNote(note, ts, sessionTs, velocity, channel, 0);
    };

    this.addNote = function(note, startTime, sessionTs, velocity, channel, duration) {

        var ts = startTime;
        if(this.offsetMusic) {
            ts = this.startTs + sessionTs;
        }

        var noteDuration = duration;
        var noteVelocity = velocity;
        var graphics;
        if(!duration) {
            if(typeof this.noteDict[channel]==='undefined'){
                this.noteDict[channel] = {};
            }
            if(velocity===0) {
                if(typeof this.noteDict[channel][note] !== 'undefined') {
                    var noteDef = this.noteDict[channel][note];
                    delete this.noteDict[channel][note];
                    noteDuration = sessionTs - noteDef.sessionTs;
                    graphics = noteDef.graphics;
                    noteVelocity = noteDef.velocity;
                    ts = noteDef.ts;
                }
            }
            else {
                noteDuration = Math.abs(Date.now() - ts);
                this.noteDict[channel][note] = { ts: ts, velocity: velocity, sessionTs: sessionTs};
            }
        }


        if(!this.offsetMusic || velocity===0) {

            var width = noteDuration * this.pixelsPerSecond / 1000;
            if(!graphics) {
                var x = (ts-this.startTs) * this.pixelsPerSecond / 1000;
                if((x+width) <  (Math.abs(this.container.x) - this.width)) {
                    // not visible. do nothing
                    return;
                }
            	if (this.dynamicRange && (this.range.bottom > note || note > this.range.top)){
            		var newScale = {};
            		newScale['bottom'] = Math.min(note, this.range.bottom);
            		newScale['top'] = Math.max(note, this.range.top);
            		this.rescaleScene(newScale);
            	}
            	var y = Math.floor(((this.range.top-this.range.bottom)-(note-this.range.bottom)+0.5) * (this.noteHeight) - (this.noteHeight/2));
                var color = this.getColor(channel);

                var alpha = (noteVelocity / 128);

                graphics = this.getNoteRect(note, x, y, color, alpha, width, this.noteHeight);
                this.container.addChild(graphics);
            }
            else {
                graphics.width = width;
            }

            if(!duration && velocity) {
                this.noteDict[channel][note].graphics = graphics;
            }
        }
    };

    //rescale scene in case a note out of range is added or a note out of the range is removed
    this.rescaleScene = function(newScale){
    	var _this = this;
    	var childrenToUpdate = [];
    	var top = this.initRange.top;
    	var bottom = this.initRange.bottom;
    	_(_this.container.children).forEach(function(child) {
    		if (typeof(child) !== 'undefined' && child.note && !isHidden(child)){
    			top = Math.max(child.note, top);
    			bottom = Math.min(child.note, bottom);
    			return childrenToUpdate.push(child);
    		}
        });
    	if (newScale){
    		this.range = newScale;
    	}else{
    		this.range.top = top;
    		this.range.bottom = bottom;
    	}
    	this.noteHeight = this.height / (this.range.top - this.range.bottom + 1);
    	childrenToUpdate.forEach(function(child) {
			child.y = Math.floor(((_this.range.top-_this.range.bottom)-(child.note-_this.range.bottom)+0.5) * (_this.noteHeight) - (_this.noteHeight/2));
			child.height = _this.noteHeight;
        });
    };

    this.addLine = function(ts){

        if(typeof(ts) === 'undefined') {
            ts = new Date();
        }
        var x = -this.container.x;
        var y = this.linesDown ? this.height - 20 : 0;

        var graphics = new PIXI.Graphics()
            .beginFill(0xFFFF00)
            .lineStyle(1, this.lineColor)
            .moveTo(0, 0)
            .lineTo(0, 20)
            .endFill();
        graphics.x = x;
        graphics.y = y;
        this.container.addChild(graphics);
        // Add text
        //var totalSec = lineNb * this.lineInterval / 1000;
        var hours = ts.getHours();
        var minutes =ts.getMinutes();
        var seconds = ts.getSeconds();
        var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds  < 10 ? '0' + seconds : seconds);

        var fontObj = { font: '10pt Arial', fill: '#444444' };
        var t = new PIXI.Text(timeStr, fontObj);
        if(isHorizontal) {
            t.x = x + 2;
            t.y = this.linesDown ? this.height - 15 : 2;
        }
        else {
            t.rotation = -Math.PI/2;
            t.x = x ;
            t.y = this.linesDown ? this.height - 2 : t.width + 2;
        }
        this.container.addChild(t);
    };

    this.moveTo = function(diffTime){
        var oldX = this.container.x;
        this.container.x = Math.floor(diffTime*this.pixelsPerSecond);
        var deltaX = Math.abs(oldX-this.container.x);
        _.forOwn(this.noteDict, function(channelDict) {
            _.forOwn(channelDict, function(noteDef) {
                if(noteDef.graphics) {
                    noteDef.graphics.width = noteDef.graphics.width + deltaX;
                }
            });
        });
    };

    this.move = function() {
        var diff = (this.startTs - Date.now())/1000;
        this.moveTo(diff);
    };

    this.removePassedObjets = function(){
        var childrenToRemove = [];
        var rescale = false;
        _(_this.container.children).forEach(function(child) {
            return typeof(child) === 'undefined' ||
                (isHidden(child) && childrenToRemove.push(child));
        });
        childrenToRemove.forEach(function(child) {
        	if (_this.dynamicRange && (_this.range.bottom === child.note || child.note === _this.range.top)){
                rescale = true;
            }
            _this.container.removeChild(child);
        });
        //externalize this test to avoid repeated call to the function rescaleScene in the previous loop
        if (rescale){
            _this.rescaleScene();
        }
    };

    this.start = function() {
        if(!started) {
            this.startTs = Date.now();
            this.addLine();
            started = true;
        }
        this.verticalLinesInterval = setInterval(function() { _this.addLine(); }, this.lineInterval);
        this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.width / this.pixelsPerSecond );
    };

    this.stop = function() {
        clearInterval(this.verticalLinesInterval);
        clearInterval(this.cleanInterval);
    };

}

module.exports = PianoRoll;

},{"lodash":"lodash","pixi":"pixi","randomColor":"randomColor"}],8:[function(require,module,exports){
/**
* scripts/stageview.js
*
* This is the starting point for your application.
* Take a look at http://browserify.org/ for more info
*/

/* global document: false */

'use strict';


var PIXI = require('pixi');
var _ = require('lodash');

var defaultConfig = {
    externalRefresh: false,
    logger: undefined,
    sceneWidth: 1024,
    sceneHeight: 768,
    framerate: 25,
    sceneBgColor: 0xFFFFFF,
    canvasContainer: 'canvasContainer',
};

function StageView(options) {

    var _this = this;
    var opts = _(options).defaults(defaultConfig).value();

    var externalRefresh = opts.externalRefresh;

    this.logger = opts.logger;
    this.framerate = opts.framerate;
    var sceneBgColor = opts.sceneBgColor;
    var sceneWidth = opts.sceneWidth;
    var sceneHeight = opts.sceneHeight;
    var canvasContainer = opts.canvasContainer;
    var timeContainer = [];
    var components = [];

    //create an new instance of a pixi stage
    this.stage = new PIXI.Stage(sceneBgColor);
    //create a renderer instance.
    var renderer = PIXI.autoDetectRenderer(sceneWidth, sceneHeight);

    this.init = function() {

        if(typeof(canvasContainer) === 'string') {
            canvasContainer = document.getElementById(canvasContainer);
        }
        if(typeof(timeContainer) === 'string') {
            timeContainer = document.getElementById(timeContainer);
        }

        canvasContainer.appendChild(renderer.view);

        components.forEach(function(c){
    		c.init();
    	});
    };

    this.registerTimeContainer = function(container) {
    	timeContainer.push(container);
    };

    this.registerComponent = function(component) {
    	components.push(component);
    	this.stage.addChild(component.container);
    };

    this.refresh = function() {
    	components.forEach(function(c){
    		c.refresh();
    	});
        renderer.render(this.stage);
    };

    // Init page and intervals
    var refreshInterval;

    this.start = function() {

        if(!externalRefresh) {
            refreshInterval = setInterval(function() {_this.refresh();}, 1000/this.framerate);
        }

        components.forEach(function(c){
    		c.start();
    	});
    };

    this.stop = function() {
        if(!externalRefresh) {
            clearInterval(refreshInterval);
        }
        
        components.forEach(function(c){
    		c.stop();
    	});
    };


    this.log = function(m) {
        if(this.logger) {
            this.logger.log(m);
        }
    };


    return this;
}

module.exports = {
    StageView: StageView
};

},{"lodash":"lodash","pixi":"pixi"}],9:[function(require,module,exports){
/**
* js/utils.js
*
* basic tools
*
*/
/*jshint bitwise: false*/
/*jshint camelcase: false */

'use strict';

var PIXI = require('pixi');
var _ = require('lodash');
var moment = require('moment');

function formatTime (ts) {
    return moment(ts).format('HH:mm:ss');
}

function colorToHex(c) {
    var m = /rgba?\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/.exec(c);
    return m ? '#' + (1 << 24 | m[1] << 16 | m[2] << 8 | m[3]).toString(16).substr(1) : c;
}

function getAnnotCategories(urlCategories, annotCategories) {

    var jsonLoader = new PIXI.JsonLoader(urlCategories, true);

    jsonLoader.on('loaded', function(res) {
        var data = res.target.json;

        while(annotCategories.length > 0) {
            annotCategories.pop();
        }

        data.sessions.forEach(function(session) {
            var annotCat = {
                ts: session.start_ts === null ? new Date(0) : Date.parse(session.start_ts),
                colors: {}
            };
            var categoriesJson = session.categories_json;
            annotCat.order = categoriesJson.order;
            if (typeof(annotCat.order['default']) === 'undefined'){
                annotCat.order.push('default');
            }
            var catList = _.clone(categoriesJson.categories);
            while(catList.length > 0) {
                var cat = catList.pop();
                if(cat.code) {
                    annotCat.colors[cat.code] = colorToHex(cat.color);
                }
                if(cat.subcategories) {
                    catList = catList.concat(cat.subcategories);
                }
            }
            categoriesJson.categories.forEach(function(cat) {
                if(cat.code) {
                    annotCat.colors[cat.code] = colorToHex(cat.color);
                }
            });
            annotCat.defaultColor = categoriesJson.defaultColor || '#536991';
            annotCategories.push(annotCat);
        });
    });

    jsonLoader.load();

}

module.exports = {
    formatTime: formatTime,
    getAnnotCategories: getAnnotCategories,
    colorToHex: colorToHex
};

},{"lodash":"lodash","moment":"moment","pixi":"pixi"}],10:[function(require,module,exports){
/**
* js/wswrapper.js
*
* simple webservice wrapper to register callbacks on onmessage
*
*/

/* global WebSocket: false */

'use strict';

function WsWrapper(wsurl, logger) {

    var url = wsurl;
    var sock = new WebSocket(url);
    var loggerObj = logger;

    var log = function(msg) {
        if(loggerObj) {
            loggerObj.log(msg);
        }
    };

    var handlers = [];

    sock.onopen = function() {
        log('Connected to ' + url);
    };

    sock.onclose = function(e) {
        log('Connection closed (wasClean = ' + e.wasClean + ', code = ' + e.code + ', reason = \'' + e.reason + '\')');
        sock = null;
    };

    sock.onmessage = function(e) {
        log('received ' + e.data);
        var data = JSON.parse(e.data);
        handlers.forEach(function(handler) {
            handler(data);
        });
    };

    this.message = function(handler) {
        if(handler) {
            handlers.push(handler);
        }
    };

}

module.exports = {
    WsWrapper: WsWrapper
};

},{}]},{},[1])(1)
});
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5vZGVfbW9kdWxlcy9icm93c2VyaWZ5L25vZGVfbW9kdWxlcy9icm93c2VyLXBhY2svX3ByZWx1ZGUuanMiLCIuL2FwcC9qcy9tYWluLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9hbm5vdHNyb2xsLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9hbm5vdHN0aW1lbGluZS5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvYW5ub3Rzdml6dmlldy5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvZG91Ymxlcm9sbC5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvbG9nZ2VyLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9waWFub3JvbGwuanMiLCIvVXNlcnMveW1oL2Rldi9wcm9qZWN0cy9tb25zL2Rldi9jbGllbnQvYW5ub3R2aXovYXBwL2pzL3N0YWdldmlldy5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvdXRpbHMuanMiLCIvVXNlcnMveW1oL2Rldi9wcm9qZWN0cy9tb25zL2Rldi9jbGllbnQvYW5ub3R2aXovYXBwL2pzL3dzd3JhcHBlci5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtBQ0FBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzlCQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQy9OQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzVRQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUN4TkE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUN6TkE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDN0NBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzFSQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDcEhBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUMxRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwiZmlsZSI6ImdlbmVyYXRlZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzQ29udGVudCI6WyIoZnVuY3Rpb24gZSh0LG4scil7ZnVuY3Rpb24gcyhvLHUpe2lmKCFuW29dKXtpZighdFtvXSl7dmFyIGE9dHlwZW9mIHJlcXVpcmU9PVwiZnVuY3Rpb25cIiYmcmVxdWlyZTtpZighdSYmYSlyZXR1cm4gYShvLCEwKTtpZihpKXJldHVybiBpKG8sITApO3ZhciBmPW5ldyBFcnJvcihcIkNhbm5vdCBmaW5kIG1vZHVsZSAnXCIrbytcIidcIik7dGhyb3cgZi5jb2RlPVwiTU9EVUxFX05PVF9GT1VORFwiLGZ9dmFyIGw9bltvXT17ZXhwb3J0czp7fX07dFtvXVswXS5jYWxsKGwuZXhwb3J0cyxmdW5jdGlvbihlKXt2YXIgbj10W29dWzFdW2VdO3JldHVybiBzKG4/bjplKX0sbCxsLmV4cG9ydHMsZSx0LG4scil9cmV0dXJuIG5bb10uZXhwb3J0c312YXIgaT10eXBlb2YgcmVxdWlyZT09XCJmdW5jdGlvblwiJiZyZXF1aXJlO2Zvcih2YXIgbz0wO288ci5sZW5ndGg7bysrKXMocltvXSk7cmV0dXJuIHN9KSIsIi8qKlxuICogc2NyaXB0cy9tYWluLmpzXG4gKlxuICogVGhpcyBpcyB0aGUgc3RhcnRpbmcgcG9pbnQgZm9yIHlvdXIgYXBwbGljYXRpb24uXG4gKiBUYWtlIGEgbG9vayBhdCBodHRwOi8vYnJvd3NlcmlmeS5vcmcvIGZvciBtb3JlIGluZm9cbiAqL1xuXG4ndXNlIHN0cmljdCc7XG5cbnZhciBkb3VibGVyb2xsID0gcmVxdWlyZSgnLi9kb3VibGVyb2xsJyk7XG52YXIgYW5ub3Rzcm9sbCA9IHJlcXVpcmUoJy4vYW5ub3Rzcm9sbCcpO1xudmFyIGFubm90c3RpbWVsaW5lID0gcmVxdWlyZSgnLi9hbm5vdHN0aW1lbGluZScpO1xudmFyIGFubm90c3ZpenZpZXcgPSByZXF1aXJlKCcuL2Fubm90c3ZpenZpZXcnKTtcbnZhciBzdGFnZXZpZXcgPSByZXF1aXJlKCcuL3N0YWdldmlldycpO1xudmFyIHdzd3JhcHBlciA9IHJlcXVpcmUoJy4vd3N3cmFwcGVyJyk7XG52YXIgbG9nZ2VyID0gcmVxdWlyZSgnLi9sb2dnZXInKTtcbnZhciB1dGlscyA9IHJlcXVpcmUoJy4vdXRpbHMnKTtcblxudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcblxubW9kdWxlLmV4cG9ydHMgPSBfKHt9KVxuICAgIC5leHRlbmQoZG91Ymxlcm9sbClcbiAgICAuZXh0ZW5kKGFubm90c3JvbGwpXG4gICAgLmV4dGVuZChhbm5vdHN0aW1lbGluZSlcbiAgICAuZXh0ZW5kKGFubm90c3ZpenZpZXcpXG4gICAgLmV4dGVuZChzdGFnZXZpZXcpXG4gICAgLmV4dGVuZCh3c3dyYXBwZXIpXG4gICAgLmV4dGVuZChsb2dnZXIpXG4gICAgLmV4dGVuZCh1dGlscylcbiAgICAudmFsdWUoKTtcbiIsIi8qKlxuKiBqcy9hbm5vdHNSb2xsLmpzXG4qXG4qIGFubm90c1JvbGwgYmFzaWMgY29tcG9uZW50XG4qXG4qL1xuXG4ndXNlIHN0cmljdCc7XG5cbnZhciBQSVhJID0gcmVxdWlyZSgncGl4aScpO1xudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcblxudmFyIERFRkFVTFRfQU5OT1RfQ09MT1IgPSAnI2JhYmFiYSc7XG5cbnZhciBkZWZhdWx0QW5ub3RTdHlsZXMgPSB7XG4gICAgJ2xhYmVsJzogeyBmb250OiAnMTZwdCBBcmlhbCBCb2xkJywgZmlsbDogJyM2NUE5NTQnLCB3b3JkV3JhcDogdHJ1ZX0sXG4gICAgJ3RleHQnIDogeyBmb250OiAnMTJwdCBBcmlhbCBSZWd1bGFyJywgZmlsbDogJyM0NDQ0NDQnLCB3b3JkV3JhcDogdHJ1ZX0sXG4gICAgJ3VzZXInIDogeyBmb250OiAnMTRwdCBBcmlhbCByZWd1bGFyJywgZmlsbDogJyM2NjY2NjYnIH0sXG59O1xuXG52YXIgZGVmYXVsdE9wdGlvbnMgPSB7XG4gICAgZXh0ZXJuYWxSZWZyZXNoOiBmYWxzZSxcbiAgICBkZWZhdWx0Q29sb3I6IERFRkFVTFRfQU5OT1RfQ09MT1IsXG4gICAgYW5ub3RTdHlsZXM6IGRlZmF1bHRBbm5vdFN0eWxlcyxcbiAgICBpZ25vcmVBbm5vdHM6ZmFsc2Vcbn07XG5cbmZ1bmN0aW9uIEFubm90c1JvbGwob3B0aW9ucykge1xuXG4vL3BhcmVudENvbnRhaW5lciwgeEluaXQsIHlJbml0LCB3aWR0aCwgaGVpZ2h0LCB3aWR0aFJvbGwsIHBpeGVsc1BlclNlY29uZCwgYW5ub3RDb2xvcnNcbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0T3B0aW9ucykudmFsdWUoKTtcblxuXG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IG9wdHMueEluaXQ7XG4gICAgdGhpcy5jb250YWluZXIueSA9IG9wdHMueUluaXQ7XG4gICAgdGhpcy5jb250YWluZXIud2lkdGggPSBvcHRzLndpZHRoO1xuXG4gICAgdGhpcy5oZWlnaHQgPSBvcHRzLmhlaWdodDtcbiAgICB0aGlzLndpZHRoID0gb3B0cy53aWR0aDtcbiAgICB0aGlzLndpZHRoUm9sbCA9IG9wdHMud2lkdGhSb2xsO1xuICAgIHRoaXMucGl4ZWxzUGVyU2Vjb25kID0gb3B0cy5waXhlbHNQZXJTZWNvbmQ7XG4gICAgdGhpcy5hbm5vdENvbG9ycyA9IG9wdHMuYW5ub3RDb2xvcnM7XG4gICAgdGhpcy5zdGFydFRzID0gb3B0cy5zdGFydFRzIHx8IERhdGUubm93KCk7XG4gICAgdGhpcy5pZ25vcmVBbm5vdHMgPSBvcHRzLmlnbm9yZUFubm90cztcblxuICAgIHZhciB5SW5pdCA9IG9wdHMueUluaXQ7XG4gICAgdmFyIGFubm90U3R5bGVzID0gXyhvcHRzLmFubm90U3R5bGVzKS5kZWZhdWx0cyhkZWZhdWx0QW5ub3RTdHlsZXMpLnZhbHVlKCk7XG4gICAgdmFyIG1hcmdpblggPSAxNTtcbiAgICBmb3IodmFyIHN0eWxlIGluIGFubm90U3R5bGVzKSB7XG4gICAgXHRpZiAoYW5ub3RTdHlsZXNbc3R5bGVdLndvcmRXcmFwID09PSB0cnVlKXtcbiAgICBcdFx0YW5ub3RTdHlsZXNbc3R5bGVdLndvcmRXcmFwV2lkdGggPSB0aGlzLndpZHRoUm9sbCAtIG1hcmdpblg7XG4gICAgXHR9XG4gICAgfVxuICAgIHZhciBzdGFydGVkID0gZmFsc2U7XG4gICAgdmFyIHdzID0gb3B0cy53cztcbiAgICB2YXIgZXh0ZXJuYWxSZWZyZXNoID0gb3B0cy5leHRlcm5hbFJlZnJlc2g7XG4gICAgdmFyIHN0YWdlVmlldyA9IG9wdHMuc3RhZ2VWaWV3O1xuICAgIHZhciB3YWl0SW50ZXJ2YWw7XG4gICAgdmFyIHdhaXQgPSAwO1xuXG4gICAgc3RhZ2VWaWV3LnJlZ2lzdGVyQ29tcG9uZW50KHRoaXMpO1xuXG4gICAgdmFyIGlzSGlkZGVuID0gZnVuY3Rpb24oY2hpbGQpIHtcbiAgICAgICAgLy8gVE9ETzogdGhlIG9yaWdpbiBwb2ludCBpcyBhbiBhcHByb3hpbWF0aW9uLiBTaG91bGQgcmVmaW5lIHRoaXNcbiAgICAgICAgdmFyIGdsb2JhbFBvcyA9IGNoaWxkLnRvR2xvYmFsKG5ldyBQSVhJLlBvaW50KDAsMCkpO1xuICAgICAgICByZXR1cm4gKChnbG9iYWxQb3MueCArIGNoaWxkLndpZHRoKSA8IDApIHx8ICgoZ2xvYmFsUG9zLnkgKyBjaGlsZC5oZWlnaHQpIDwgMCkgO1xuICAgIH07XG5cbiAgICB0aGlzLmFkZEFubm90cyA9IGZ1bmN0aW9uKGRhdGEpIHtcblxuICAgICAgICAvL3ZhciB0aXRsZSA9IGRhdGEuY29udGVudC5jYXRlZ29yeS5sYWJlbDtcbiAgICAgICAgLy92YXIgdXNlciA9IGRhdGEuY29udGVudC51c2VyO1xuICAgICAgICAvL1Rlc3QgY2F0IGFuZCBjb2xvclxuICAgICAgICAvL3ZhciBjb2xvckFubm90ID0gMHg2NUE5NTQ7XG4gICAgICAgIHZhciBjYXRlZ29yeSA9IGRhdGEuY29udGVudC5jYXRlZ29yeS5sYWJlbCxcbiAgICAgICAgICAgIHRleHQgICAgID0gZGF0YS5jb250ZW50LnRleHQsXG4gICAgICAgICAgICB1c2VyICAgICA9IGRhdGEuY29udGVudC51c2VyLFxuICAgICAgICAgICAgdHMgICAgICAgPSBEYXRlLnBhcnNlKGRhdGEudHMpLFxuICAgICAgICAgICAgY29sb3IgICAgPSBkYXRhLmNvbnRlbnQuY29sb3IgfHwgdGhpcy5nZXRDb2xvcih0cywgZGF0YS5jb250ZW50LmNhdGVnb3J5LmNvZGUpO1xuXG4gICAgICAgIHRoaXMuYWRkQW5ub3QoY2F0ZWdvcnksIHRleHQsIHVzZXIsIGNvbG9yLCB0cyk7XG4gICAgfTtcblxuICAgIHRoaXMuZ2V0Q29sb3IgPSBmdW5jdGlvbih0cywgY29kZSkge1xuICAgICAgICB2YXIgY29sb3JzRGVmO1xuICAgICAgICBfKHRoaXMuYW5ub3RDb2xvcnMpLmVhY2hSaWdodChmdW5jdGlvbihjZGVmKSB7XG4gICAgICAgICAgICBpZihjZGVmLnRzIDwgdHMpIHtcbiAgICAgICAgICAgICAgICBjb2xvcnNEZWYgPSBjZGVmLmNvbG9ycztcbiAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgICAgICB2YXIgcmVzQ29sb3I7XG4gICAgICAgIGlmKGNvbG9yc0RlZikge1xuICAgICAgICAgICAgcmVzQ29sb3IgPSBjb2xvcnNEZWZbY29kZV07XG4gICAgICAgIH1cbiAgICAgICAgaWYoIXJlc0NvbG9yKSB7XG4gICAgICAgICAgICByZXNDb2xvciA9IGNvbG9yc0RlZi5kZWZhdWx0Q29sb3IgfHwgREVGQVVMVF9BTk5PVF9DT0xPUjtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gcmVzQ29sb3I7XG4gICAgfTtcblxuICAgIHRoaXMuYWRkQW5ub3QgPSBmdW5jdGlvbihjYXRlZ29yeSwgdGV4dCwgdXNlciwgY29sb3IsIHRzKXtcblxuICAgICAgICB2YXIgeCA9IDA7XG4gICAgICAgIHZhciB5ID0gKHRzLXRoaXMuc3RhcnRUcykgKiB0aGlzLnBpeGVsc1BlclNlY29uZCAvIDEwMDAgKyB5SW5pdDtcblxuICAgICAgICB2YXIgY29sb3JIZXggPSBwYXJzZUludChjb2xvci5yZXBsYWNlKC9eIy8sICcnKSwgMTYpO1xuXG4gICAgICAgIGlmICh3YWl0ID09PSAwKXtcblx0ICAgICAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpXG5cdCAgICAgICAgICAgIC5iZWdpbkZpbGwoY29sb3JIZXgpXG5cdCAgICAgICAgICAgIC5kcmF3UmVjdCh4LCB5LCAxMCwgMylcblx0ICAgICAgICAgICAgLmVuZEZpbGwoKTtcblxuXHQgICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGdyYXBoaWNzKTtcblxuXHQgICAgICAgIHZhciB0ZXh0SGVpZ2h0ID0gMDtcblx0ICAgICAgICB2YXIgY2F0TGFiZWwgPSBuZXcgUElYSS5UZXh0KFxuXHQgICAgICAgICAgICBjYXRlZ29yeSxcblx0ICAgICAgICAgICAgXyhhbm5vdFN0eWxlcy5sYWJlbCkuZXh0ZW5kKHtmaWxsOiBjb2xvcn0pLnZhbHVlKClcblx0ICAgICAgICApO1xuXHQgICAgICAgIGNhdExhYmVsLnggPSB4ICsgbWFyZ2luWDtcblx0ICAgICAgICBjYXRMYWJlbC55ID0geSAtIDIzO1xuXHQgICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGNhdExhYmVsKTtcblx0ICAgICAgICB0ZXh0SGVpZ2h0ICs9IChjYXRMYWJlbC5oZWlnaHQgLSAyMyArIDIpO1xuXG5cdCAgICAgICAgaWYodGV4dCkge1xuXHQgICAgICAgICAgICB2YXIgY2F0VGV4dCA9IG5ldyBQSVhJLlRleHQodGV4dCwgYW5ub3RTdHlsZXMudGV4dCk7XG5cdCAgICAgICAgICAgIGNhdFRleHQueCA9IHggKyBtYXJnaW5YO1xuXHQgICAgICAgICAgICBjYXRUZXh0LnkgPSB5ICsgdGV4dEhlaWdodDtcblx0ICAgICAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY2F0VGV4dCk7XG5cdCAgICAgICAgICAgIHRleHRIZWlnaHQgKz0gKGNhdFRleHQuaGVpZ2h0ICsgMik7XG5cdCAgICAgICAgfVxuXG5cdCAgICAgICAgdmFyIGNhdFVzZXIgPSBuZXcgUElYSS5UZXh0KHVzZXIsIGFubm90U3R5bGVzLnVzZXIpO1xuXHQgICAgICAgIGNhdFVzZXIueCA9IHggKyBtYXJnaW5YO1xuXHQgICAgICAgIGNhdFVzZXIueSA9IHkgKyB0ZXh0SGVpZ2h0O1xuXHQgICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGNhdFVzZXIpO1xuXHQgICAgICAgIHRleHRIZWlnaHQgKz0gKGNhdFVzZXIuaGVpZ2h0ICsgOCk7XG5cblx0ICAgICAgICBpZiAodGhpcy5pZ25vcmVBbm5vdHMgPT09IHRydWUpe1xuXHRcdCAgICAgICAgd2FpdCA9IHRleHRIZWlnaHQgLyB0aGlzLnBpeGVsc1BlclNlY29uZDtcblx0XHQgICAgICAgIHdhaXRJbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uKCkge190aGlzLnJlZnJlc2hXYWl0KCk7fSwgMTAwMCk7XG5cdCAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgdGhpcy5hZGRBbm5vdExpbmUoY29sb3JIZXgsIHkpO1xuICAgIH07XG5cbiAgICB0aGlzLmFkZEFubm90TGluZSA9IGZ1bmN0aW9uKGNvbG9yLCB5KSB7XG4gICAgICAgIHZhciB4ID0gdGhpcy53aWR0aFJvbGw7XG5cblxuICAgICAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpXG4gICAgICAgICAgICAuYmVnaW5GaWxsKGNvbG9yKVxuICAgICAgICAgICAgLmRyYXdSZWN0KHgsIHksIHRoaXMud2lkdGggLSB4LCAzKVxuICAgICAgICAgICAgLmVuZEZpbGwoKTtcblxuICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChncmFwaGljcyk7XG4gICAgfTtcblxuICAgIHRoaXMubW92ZVRvID0gZnVuY3Rpb24oZGlmZlRpbWUpe1xuICAgIFx0dGhpcy5jb250YWluZXIueSA9IE1hdGguZmxvb3IoZGlmZlRpbWUqdGhpcy5waXhlbHNQZXJTZWNvbmQpO1xuICAgIH07XG5cbiAgICB0aGlzLm1vdmUgPSB0aGlzLnJlZnJlc2ggPSBmdW5jdGlvbigpIHtcbiAgICAgICAgdmFyIGRpZmYgPSAodGhpcy5zdGFydFRzIC0gRGF0ZS5ub3coKSkvMTAwMDtcbiAgICAgICAgdGhpcy5tb3ZlVG8oZGlmZik7XG4gICAgfTtcblxuICAgIHRoaXMucmVmcmVzaFdhaXQgPSBmdW5jdGlvbigpe1xuICAgIFx0d2FpdCAtPSAxO1xuICAgIFx0aWYgKHdhaXQgPCAwKXtcbiAgICBcdFx0d2FpdCA9IDA7XG4gICAgXHRcdGNsZWFySW50ZXJ2YWwod2FpdEludGVydmFsKTtcbiAgICBcdH1cbiAgICB9O1xuXG4gICAgdGhpcy5yZW1vdmVQYXNzZWRPYmpldHMgPSBmdW5jdGlvbigpe1xuICAgICAgICB2YXIgY2hpbGRyZW5Ub1JlbW92ZSA9IFtdO1xuICAgICAgICBfKF90aGlzLmNvbnRhaW5lci5jaGlsZHJlbikuZm9yRWFjaChmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICAgICAgcmV0dXJuIHR5cGVvZihjaGlsZCkgPT09ICd1bmRlZmluZWQnIHx8XG4gICAgICAgICAgICAgICAgKGlzSGlkZGVuKGNoaWxkKSAmJiBjaGlsZHJlblRvUmVtb3ZlLnB1c2goY2hpbGQpKTtcbiAgICAgICAgfSk7XG4gICAgICAgIGNoaWxkcmVuVG9SZW1vdmUuZm9yRWFjaChmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICAgICAgX3RoaXMuY29udGFpbmVyLnJlbW92ZUNoaWxkKGNoaWxkKTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIHRoaXMuaW5pdCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgICAgIHdzLm1lc3NhZ2UoZnVuY3Rpb24oZGF0YSkge1xuICAgICAgICAgICAgX3RoaXMuYWRkQW5ub3RzKGRhdGEpO1xuICAgICAgICB9KTtcblxuICAgIH07XG5cblxuICAgIHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgaWYoIXN0YXJ0ZWQpIHtcbiAgICAgICAgICAgIHRoaXMuc3RhcnRUcyA9IERhdGUubm93KCk7XG4gICAgICAgICAgICBzdGFydGVkID0gdHJ1ZTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLmNsZWFuSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbiAoKSB7IF90aGlzLnJlbW92ZVBhc3NlZE9iamV0cygpOyB9LCAxMDAwICogdGhpcy5oZWlnaHQgLyB0aGlzLnBpeGVsc1BlclNlY29uZCApO1xuICAgICAgICBpZighZXh0ZXJuYWxSZWZyZXNoKSB7XG4gICAgICAgICAgICB0aGlzLnJlZnJlc2hJbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uKCkge190aGlzLm1vdmUoKTt9LCAxMDAwL3RoaXMuZnJhbWVyYXRlKTtcbiAgICAgICAgfVxuICAgIH07XG5cbiAgICB0aGlzLnN0b3AgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLmNsZWFuSW50ZXJ2YWwpO1xuICAgICAgICBpZighZXh0ZXJuYWxSZWZyZXNoKSB7XG4gICAgICAgICAgICBjbGVhckludGVydmFsKHRoaXMucmVmcmVzaEludGVydmFsKTtcbiAgICAgICAgfVxuICAgIH07XG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgQW5ub3RzUm9sbDogQW5ub3RzUm9sbCxcbn07XG4iLCIvKipcbioganMvYW5ub3RzdGltZWxpbmVcbipcbiogYW5ub3RzdGltZWxpbmUgYmFzaWMgY29tcG9uZW50XG4qXG4qL1xuXG4ndXNlIHN0cmljdCc7XG5cbnZhciBQSVhJID0gcmVxdWlyZSgncGl4aScpO1xudmFyIFV0aWxzID0gcmVxdWlyZSgnLi91dGlscy5qcycpO1xudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcblxudmFyIGRlZmF1bHRPcHRpb25zID0ge1xuICAgIGxvZ2dlcjogdW5kZWZpbmVkLFxuICAgIGludGVydmFsV2lkdGg6IDEwLFxuICAgIGludGVydmFsSGVpZ2h0OiA1LFxuICAgIG1heENlbGxIZWlnaHQ6IDIwMCxcbiAgICByYWRpdXM6IDMwMCxcbiAgICBzZXJ2ZXJVcmw6ICdodHRwOi8vMTI3LjAuMC4xOjgwODAnLFxuICAgIGNoYW5uZWw6ICdBTk5PVCcsXG4gICAgbWF4UGFnZXM6IDEwMDAsXG4gICAgc2hvd0Nsb2NrR3JhcGhpY3M6IHRydWUsXG4gICAgYXJjaGl2ZTogZmFsc2Vcbn07XG5cblxuZnVuY3Rpb24gQW5ub3RzVGltZUxpbmUob3B0aW9ucyl7XG4gICAgdmFyIF90aGlzID0gdGhpcztcbiAgICB2YXIgb3B0cyA9IF8ob3B0aW9ucykuZGVmYXVsdHMoZGVmYXVsdE9wdGlvbnMpLnZhbHVlKCk7XG5cbiAgICB0aGlzLmNvbnRhaW5lciA9IG5ldyBQSVhJLkRpc3BsYXlPYmplY3RDb250YWluZXIoKTtcbiAgICB0aGlzLmNvbnRhaW5lci54ID0gb3B0cy54SW5pdDtcbiAgICB0aGlzLmNvbnRhaW5lci55ID0gb3B0cy55SW5pdDtcbiAgICB0aGlzLmNvbnRhaW5lci53aWR0aCA9IG9wdHMud2lkdGg7XG4gICAgdGhpcy5jb250YWluZXIuaGVpZ2h0ID0gb3B0cy5oZWlnaHQ7XG5cbiAgICB0aGlzLnRpbWVCZWdpbiA9IG9wdHMudGltZUJlZ2luO1xuICAgIHRoaXMudGltZUVuZCA9IG9wdHMudGltZUVuZDtcbiAgICB0aGlzLmR1cmF0aW9uID0gKHRoaXMudGltZUVuZCAtIHRoaXMudGltZUJlZ2luKS8xMDAwO1xuICAgIHRoaXMud2lkdGggPSBvcHRzLndpZHRoO1xuICAgIHRoaXMuaGVpZ2h0ID0gb3B0cy5oZWlnaHQ7XG4gICAgdGhpcy5pbnRlcnZhbEhlaWdodCA9IG9wdHMuaW50ZXJ2YWxIZWlnaHQ7XG4gICAgdGhpcy5pbnRlcnZhbFdpZHRoID0gb3B0cy5pbnRlcnZhbFdpZHRoO1xuICAgIHRoaXMubWF4Q2VsbEhlaWdodCA9IG9wdHMubWF4Q2VsbEhlaWdodDtcbiAgICB0aGlzLmFubm90Q2F0ZWdvcmllcyA9IG9wdHMuYW5ub3RDYXRlZ29yaWVzO1xuICAgIHRoaXMuc3RhcnRUcyA9IG9wdGlvbnMuc3RhcnRUcyB8fCBEYXRlLm5vdygpO1xuICAgIHRoaXMuc2hvd0Nsb2NrR3JhcGhpY3MgPSBvcHRzLnNob3dDbG9ja0dyYXBoaWNzO1xuICAgIHRoaXMuYXJjaGl2ZSA9IG9wdHMuYXJjaGl2ZTtcblxuICAgIHRoaXMuY2lyY2xlWCA9IG9wdHMuY2lyY2xlWCB8fCAodGhpcy53aWR0aC8yKTtcbiAgICB0aGlzLmNpcmNsZVkgPSBvcHRzLmNpcmNsZVkgfHwgKHRoaXMuaGVpZ2h0LzIpO1xuICAgIHRoaXMucmFkaXVzID0gb3B0cy5yYWRpdXM7XG4gICAgdmFyIHBlcmltZXRlciA9IDIqTWF0aC5QSSogdGhpcy5yYWRpdXM7XG4gICAgdGhpcy5pbnRlcnZhbER1cmF0aW9uID0gKHRoaXMuaW50ZXJ2YWxXaWR0aCAqIHRoaXMuZHVyYXRpb24gLyBwZXJpbWV0ZXIpO1xuXG4gICAgdmFyIGNoYW5uZWwgPSBvcHRzLmNoYW5uZWw7XG4gICAgdmFyIGV2ZW50Q29kZSA9IG9wdHMuZXZlbnRDb2RlO1xuICAgIHZhciBzZXJ2ZXJVcmwgPSBvcHRzLnNlcnZlclVybDtcbiAgICB2YXIgbWF4UGFnZXMgPSBvcHRzLm1heFBhZ2VzO1xuXG4gICAgdmFyIHRvdGFsSW5kZXggPSBNYXRoLmZsb29yKCBwZXJpbWV0ZXIvdGhpcy5pbnRlcnZhbFdpZHRoKTtcblxuICAgIHRoaXMuY2VsbHMgPSBbXTtcbiAgICBmb3IgKHZhciBpPTA7IGk8KHBlcmltZXRlci90aGlzLmludGVydmFsV2lkdGgpIDsgaSsrKXtcbiAgICAgICAgdGhpcy5jZWxsc1tpXSA9IFtdO1xuICAgICAgICB0aGlzLmNlbGxzW2ldLmkgPSBpO1xuICAgICAgICB0aGlzLmNlbGxzW2ldLnRvdGFsQW5ub3RzID0gMDtcbiAgICAgICAgdGhpcy5jZWxsc1tpXS5jYXRlZ29yaWVzID0ge307XG4gICAgfVxuXG4gICAgdmFyIHdzID0gb3B0cy53cztcbiAgICB2YXIgc3RhZ2VWaWV3ID0gb3B0cy5zdGFnZVZpZXc7XG5cbiAgICAvL2RyYXcgdGhlIGJhc2UgLSBjaXJjbGUgYW5kIGxpbmUgdG8gbG9jYXRlIHRoZSBzY2VuZVxuICAgIHZhciBncmFwaGljcyA9IG5ldyBQSVhJLkdyYXBoaWNzKCk7XG4gICAgZ3JhcGhpY3MubGluZVN0eWxlKDIsIDB4NjQ2NDY0KVxuICAgICAgICAuZHJhd0NpcmNsZSh0aGlzLmNpcmNsZVgsIHRoaXMuY2lyY2xlWSwgdGhpcy5yYWRpdXMgLSAzKVxuICAgICAgICAuZW5kRmlsbCgpO1xuICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGdyYXBoaWNzKTtcblxuICAgIHN0YWdlVmlldy5yZWdpc3RlckNvbXBvbmVudCh0aGlzKTtcblxuICAgIHZhciBsb2FkQXJjaGl2ZXMgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgLy9zdGFydCB0aW1lQmVnaW4gZW5kIHN0YXJ0VGltZVxuICAgICAgICAvL3F1ZXJ5IC0+IG5lZWQgY2hhbm5lbCArIGV2ZW50Q29kZVxuICAgICAgICAvL2l0ZXJhdGUgb3ZlciBkYXRhIGZpbGwgY2VsbHNcbiAgICAgICAgdmFyIHN0YXJ0VHMgPSBfdGhpcy50aW1lQmVnaW47XG4gICAgICAgIHZhciBlbmRUcyA9IE1hdGgubWluKF90aGlzLnRpbWVFbmQsX3RoaXMuc3RhcnRUcyk7XG5cbiAgICAgICAgdmFyIHVybCA9IHNlcnZlclVybCArICcvcC9hcGkvdjEvYW5ub3RhdGlvbic7XG4gICAgICAgIHZhciBmaWx0ZXJzID0gW1xuICAgICAgICAgICAgeyBuYW1lOiAndHMnLCBvcDogJz4nLCB2YWw6IG5ldyBEYXRlKHN0YXJ0VHMpLnRvSVNPU3RyaW5nKCl9LCAvL3N0YXJ0XG4gICAgICAgICAgICB7IG5hbWU6ICd0cycsIG9wOiAnPD0nLCB2YWw6IG5ldyBEYXRlKGVuZFRzKS50b0lTT1N0cmluZygpfSwgLy9lbmRcbiAgICAgICAgICAgIHsgbmFtZTogJ2NoYW5uZWwnLCBvcDogJz09JywgdmFsOiBjaGFubmVsfSwgLy9jaGFubmVsXG4gICAgICAgICAgICB7IG5hbWU6ICdldmVudF9jb2RlJywgb3A6ICc9PScsIHZhbDogZXZlbnRDb2RlfSAvL2V2ZW50Y29kZVxuICAgICAgICBdO1xuXG4gICAgICAgIHVybCA9IHVybCArICc/cT0nICsgSlNPTi5zdHJpbmdpZnkoe2ZpbHRlcnM6ZmlsdGVyc30pO1xuXG4gICAgICAgIHZhciB0b3RhbFBhZ2UgPSAxO1xuICAgICAgICB2YXIgY3VycmVudFBhZ2UgPSAxO1xuXG4gICAgICAgIHZhciBwcm9jZXNzUmVzRnVuY3Rpb24gPSBmdW5jdGlvbihyZXMpIHtcbiAgICAgICAgICAgIC8vY29uc29sZS5sb2coXCJSRVMgYXJjaGl2ZVwiLCByZXMpO1xuICAgICAgICAgICAgaWYocmVzKSB7XG4gICAgICAgICAgICAgICAgdmFyIGRhdGEgPSByZXMudGFyZ2V0Lmpzb247XG4gICAgICAgICAgICAgICAgLypqc2hpbnQgLVcwNjkgKi9cbiAgICAgICAgICAgICAgICB0b3RhbFBhZ2UgPSBNYXRoLm1pbihtYXhQYWdlcyxwYXJzZUludChkYXRhWyd0b3RhbF9wYWdlcyddKSk7XG4gICAgICAgICAgICAgICAgZGF0YS5vYmplY3RzLmZvckVhY2goZnVuY3Rpb24oYW5ub3RhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICBfdGhpcy5hZGRBbm5vdChhbm5vdGF0aW9uKTtcbiAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGlmKGN1cnJlbnRQYWdlIDw9IHRvdGFsUGFnZSkge1xuICAgICAgICAgICAgICAgIHZhciBqc29uTG9hZGVyID0gbmV3IFBJWEkuSnNvbkxvYWRlcih1cmwrJyZwYWdlPScrY3VycmVudFBhZ2UsIHRydWUpO1xuICAgICAgICAgICAgICAgIGpzb25Mb2FkZXIub24oJ2xvYWRlZCcsIHByb2Nlc3NSZXNGdW5jdGlvbik7XG4gICAgICAgICAgICAgICAganNvbkxvYWRlci5sb2FkKCk7XG4gICAgICAgICAgICAgICAgY3VycmVudFBhZ2UrKztcbiAgICAgICAgICAgIH1cbiAgICAgICAgfTtcbiAgICAgICAgcHJvY2Vzc1Jlc0Z1bmN0aW9uKCk7XG5cbiAgICB9O1xuXG4gICAgLy9BZGQgQW5ub3RhdGlvbiB0byB0aGUgVGltZUxpbmVcbiAgICB0aGlzLmFkZEFubm90ID0gZnVuY3Rpb24oZGF0YSl7XG5cbiAgICAgICAgdmFyIHRzID0gRGF0ZS5wYXJzZShkYXRhLnRzKTtcbiAgICAgICAgdmFyIGNvbG9yc0RlZjtcbiAgICAgICAgXyh0aGlzLmFubm90Q2F0ZWdvcmllcykuZWFjaFJpZ2h0KGZ1bmN0aW9uKGNkZWYpIHtcbiAgICAgICAgICAgIGlmKGNkZWYudHMgPCB0cykge1xuICAgICAgICAgICAgICAgIGNvbG9yc0RlZiA9IGNkZWY7XG4gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICAgICAgfVxuICAgICAgICB9KTtcblxuICAgICAgICBpZighY29sb3JzRGVmKSB7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBpZiAodGhpcy50aW1lRW5kID4gdHMpe1xuICAgICAgICAgICAgdmFyIGkgPSBNYXRoLmZsb29yKCh0cyAtIHRoaXMudGltZUJlZ2luKS8oMTAwMCp0aGlzLmludGVydmFsRHVyYXRpb24pKTtcblxuICAgICAgICAgICAgaWYgKHR5cGVvZih0aGlzLmNlbGxzW2ldLmdyYXBoaWNzKSA9PT0gJ3VuZGVmaW5lZCcpe1xuICAgICAgICAgICAgICAgIHRoaXMuaW5pdENlbGwodGhpcy5jZWxsc1tpXSwgY29sb3JzRGVmKTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgdmFyIGFubm90Q29kZTtcbiAgICAgICAgICAgIGlmICh0eXBlb2YoY29sb3JzRGVmLmNvbG9yc1tkYXRhLmNvbnRlbnQuY2F0ZWdvcnkuY29kZV0pICE9PSAndW5kZWZpbmVkJyl7XG4gICAgICAgICAgICAgICAgYW5ub3RDb2RlID0gZGF0YS5jb250ZW50LmNhdGVnb3J5LmNvZGU7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIGFubm90Q29kZSA9ICdkZWZhdWx0JztcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgdGhpcy5jZWxsc1tpXS5jYXRlZ29yaWVzW2Fubm90Q29kZV0uY291bnQgKz0gMTtcbiAgICAgICAgICAgIHRoaXMuY2VsbHNbaV0udG90YWxBbm5vdHMgKz0xO1xuICAgICAgICAgICAgdGhpcy5yZWRyYXdDZWxsKHRoaXMuY2VsbHNbaV0sIGNvbG9yc0RlZik7XG4gICAgICAgIH1cbiAgICB9O1xuXG4gICAgdGhpcy5pbml0Q2xvY2tHcmFwaGljcyA9IGZ1bmN0aW9uKCkge1xuICAgICAgICB2YXIgdEJlZyA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSh0aGlzLnRpbWVCZWdpbiksIHsgZm9udDogJzEycHQgR290aGljIFN0YW5kYXJkJywgZmlsbDogJyM2NDY0NjQnIH0pO1xuICAgICAgICB0QmVnLnggPSB0aGlzLmNpcmNsZVggKyAxNTtcbiAgICAgICAgdEJlZy55ID0gdGhpcy5jaXJjbGVZIC0gdGhpcy5yYWRpdXMgLSB0aGlzLm1heENlbGxIZWlnaHQgLSAxMDtcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodEJlZyk7XG5cbiAgICAgICAgdmFyIHRFbmQgPSBuZXcgUElYSS5UZXh0KFV0aWxzLmZvcm1hdFRpbWUodGhpcy50aW1lRW5kKSwgeyBmb250OiAnMTJwdCBHb3RoaWMgU3RhbmRhcmQnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG4gICAgICAgIHRFbmQueCA9IHRoaXMuY2lyY2xlWCAtIDE1IC0gdEVuZC53aWR0aDtcbiAgICAgICAgdEVuZC55ID0gdGhpcy5jaXJjbGVZIC0gdGhpcy5yYWRpdXMgLSB0aGlzLm1heENlbGxIZWlnaHQgLSAxMDtcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodEVuZCk7XG5cbiAgICAgICAgdmFyIHQxNSA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSgoKHRoaXMudGltZUVuZCAtIHRoaXMudGltZUJlZ2luKS80KSArIHRoaXMudGltZUJlZ2luKSwgeyBmb250OiAnMTJwdCBHb3RoaWMgU3RhbmRhcmQnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG4gICAgICAgIHQxNS54ID0gdGhpcy5jaXJjbGVYICsgdGhpcy5yYWRpdXMgKyB0aGlzLm1heENlbGxIZWlnaHQgKyAxMCA7XG4gICAgICAgIHQxNS55ID0gdGhpcy5jaXJjbGVZIC0gdDE1LmhlaWdodDtcbiAgICAgICAgdDE1LnJvdGF0aW9uID0gTWF0aC5QSSAvMjtcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodDE1KTtcblxuICAgICAgICB2YXIgdDMwID0gbmV3IFBJWEkuVGV4dChVdGlscy5mb3JtYXRUaW1lKCgodGhpcy50aW1lRW5kIC0gdGhpcy50aW1lQmVnaW4pLzIpICsgdGhpcy50aW1lQmVnaW4pLCB7IGZvbnQ6ICcxMnB0IEdvdGhpYyBTdGFuZGFyZCcsIGZpbGw6ICcjNjQ2NDY0JyB9KTtcbiAgICAgICAgdDMwLnggPSB0aGlzLmNpcmNsZVggLSB0MzAud2lkdGgvMjtcbiAgICAgICAgdDMwLnkgPSB0aGlzLmNpcmNsZVkgKyB0aGlzLnJhZGl1cyArIHRoaXMubWF4Q2VsbEhlaWdodCAtIDI7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKHQzMCk7XG5cbiAgICAgICAgdmFyIHQ0NSA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSgoKHRoaXMudGltZUVuZCAtIHRoaXMudGltZUJlZ2luKSozLzQpICsgdGhpcy50aW1lQmVnaW4pLCB7IGZvbnQ6ICcxMnB0IEdvdGhpYyBTdGFuZGFyZCcsIGZpbGw6ICcjNjQ2NDY0JyB9KTtcbiAgICAgICAgdDQ1LnggPSB0aGlzLmNpcmNsZVggLSB0aGlzLnJhZGl1cyAtIHRoaXMubWF4Q2VsbEhlaWdodCAtIDEwIDtcbiAgICAgICAgdDQ1LnkgPSB0aGlzLmNpcmNsZVkgKyB0MTUuaGVpZ2h0O1xuICAgICAgICB0NDUucm90YXRpb24gPSAtTWF0aC5QSS8yO1xuICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZCh0NDUpO1xuXG4gICAgICAgIHZhciBsaW5lViA9IG5ldyBQSVhJLkdyYXBoaWNzKCk7XG4gICAgICAgIGxpbmVWLmxpbmVTdHlsZSgxLCAweDY0NjQ2NClcbiAgICAgICAgICAgIC5tb3ZlVG8odGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVkgLSAodGhpcy5yYWRpdXMvMykvMilcbiAgICAgICAgICAgIC5saW5lVG8odGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVkgLSB0aGlzLnJhZGl1cyAtIHRoaXMubWF4Q2VsbEhlaWdodCAtIDEwKVxuICAgICAgICAgICAgLmVuZEZpbGwoKTtcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQobGluZVYpO1xuICAgIH07XG5cbiAgICAvL0RyYXcgdGhlIGNlbGx1bGVcbiAgICB0aGlzLnJlZHJhd0NlbGwgPSBmdW5jdGlvbihjZWxsLCBjb2xvcnNEZWYpe1xuICAgICAgICB2YXIgeSA9IDA7XG5cbiAgICAgICAgLy9DaGVjayBpZiB0b3RhbCBoZWlnaHQgaXMgaGlnaGVyIHRoYW4gTWF4IENlbGwgSGVpZ2h0XG4gICAgICAgIHZhciBoZWlnaHRTdGVwO1xuICAgICAgICBpZiAoKGNlbGwudG90YWxBbm5vdHMqdGhpcy5pbnRlcnZhbEhlaWdodCkgPiB0aGlzLm1heENlbGxIZWlnaHQpe1xuICAgICAgICAgICAgaGVpZ2h0U3RlcCA9IHRoaXMubWF4Q2VsbEhlaWdodC9jZWxsLnRvdGFsQW5ub3RzO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgaGVpZ2h0U3RlcCA9IHRoaXMuaW50ZXJ2YWxIZWlnaHQ7XG4gICAgICAgIH1cblxuICAgICAgICAvL0RyYXcgdGhlIHJlY3QgZGVwZW5kaW5nIG9uIHRoZSBoZWlnaHQgc3RlcCBjYWxjdWxhdGVkXG4gICAgICAgIGZvciAodmFyIGk9MDsgaTwgY29sb3JzRGVmLm9yZGVyLmxlbmd0aDsgaSsrKXtcbiAgICAgICAgICAgIHZhciBjdXJyZW50Q29kZSA9IGNvbG9yc0RlZi5vcmRlcltpXTtcbiAgICAgICAgICAgIGNlbGwuZ3JhcGhpY3MuYmVnaW5GaWxsKGNlbGwuY2F0ZWdvcmllc1tjdXJyZW50Q29kZV0uY29sb3IucmVwbGFjZSgnIycsICcweCcpKVxuICAgICAgICAgICAgICAgIC5kcmF3UmVjdCgwLCB5LCB0aGlzLmludGVydmFsV2lkdGgtMSwgLWNlbGwuY2F0ZWdvcmllc1tjdXJyZW50Q29kZV0uY291bnQgKiBoZWlnaHRTdGVwKVxuICAgICAgICAgICAgICAgIC5lbmRGaWxsKCk7XG4gICAgICAgICAgICB5IC09IGNlbGwuY2F0ZWdvcmllc1tjdXJyZW50Q29kZV0uY291bnQqaGVpZ2h0U3RlcDtcbiAgICAgICAgfVxuICAgIH07XG5cbiAgICB0aGlzLmluaXRDZWxsID0gZnVuY3Rpb24oY2VsbCwgY29sb3JzRGVmKXtcbiAgICAgICAgY2VsbC5ncmFwaGljcyA9IG5ldyBQSVhJLkdyYXBoaWNzKCk7XG4gICAgICAgIGNlbGwuZ3JhcGhpY3MucG9zaXRpb24ueCA9IHRoaXMuY2lyY2xlWCArIHRoaXMucmFkaXVzICogTWF0aC5zaW4oY2VsbC5pKigzNjAvdG90YWxJbmRleCkqKE1hdGguUEkvMTgwKSk7XG4gICAgICAgIGNlbGwuZ3JhcGhpY3MucG9zaXRpb24ueSA9IHRoaXMuY2lyY2xlWSAtIHRoaXMucmFkaXVzICogTWF0aC5jb3MoY2VsbC5pKigzNjAvdG90YWxJbmRleCkqKE1hdGguUEkvMTgwKSk7XG4gICAgICAgIGNlbGwuZ3JhcGhpY3Mucm90YXRpb24gPSAoY2VsbC5pKSooMzYwL3RvdGFsSW5kZXgpKihNYXRoLlBJLzE4MCkgKyAoMzYwLyh0b3RhbEluZGV4KjIpKSooTWF0aC5QSS8xODApO1xuICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChjZWxsLmdyYXBoaWNzKTtcblxuICAgICAgICBmb3IgKHZhciBjYXRlZ29yeSBpbiBjb2xvcnNEZWYuY29sb3JzKXtcbiAgICAgICAgICAgIGNlbGwuY2F0ZWdvcmllc1tjYXRlZ29yeV0gPSB7XG4gICAgICAgICAgICAgICAgJ2NvdW50JzogMCxcbiAgICAgICAgICAgICAgICAnY29sb3InOiBjb2xvcnNEZWYuY29sb3JzW2NhdGVnb3J5XVxuICAgICAgICAgICAgfTtcbiAgICAgICAgfVxuICAgICAgICBpZiAodHlwZW9mKGNlbGwuY2F0ZWdvcmllc1snZGVmYXVsdCddKSA9PT0gJ3VuZGVmaW5lZCcpe1xuICAgICAgICAgICAgY2VsbC5jYXRlZ29yaWVzWydkZWZhdWx0J10gPSB7XG4gICAgICAgICAgICAgICAgJ2NvdW50JzogMCxcbiAgICAgICAgICAgICAgICAnY29sb3InOiBjb2xvcnNEZWYuZGVmYXVsdENvbG9yXG4gICAgICAgICAgICB9O1xuICAgICAgICB9XG4gICAgfTtcblxuICAgIHRoaXMuaW5pdCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBpZiAoIXRoaXMuYXJjaGl2ZSl7XG4gICAgICAgICAgICB3cy5tZXNzYWdlKGZ1bmN0aW9uKGRhdGEpIHtcbiAgICAgICAgICAgICAgICBfdGhpcy5hZGRBbm5vdChkYXRhKTtcbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHRoaXMuc2hvd0Nsb2NrR3JhcGhpY3Mpe3RoaXMuaW5pdENsb2NrR3JhcGhpY3MoKTt9XG4gICAgfTtcblxuXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICB0aGlzLnN0YXJ0VHMgPSBEYXRlLm5vdygpO1xuICAgICAgICBsb2FkQXJjaGl2ZXMoKTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG5cbiAgICB9O1xuXG4gICAgdGhpcy5zdG9wID0gZnVuY3Rpb24oKXtcbiAgICB9O1xuXG4gICAgcmV0dXJuIHRoaXM7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIEFubm90c1RpbWVMaW5lOiBBbm5vdHNUaW1lTGluZVxufTtcbiIsIi8qKlxuKiBqcy9hbm5vdHN2aXp2aWV3LmpzXG4qXG4qIFRoaXMgaXMgdGhlIHN0YXJ0aW5nIHBvaW50IGZvciB5b3VyIGFwcGxpY2F0aW9uLlxuKiBUYWtlIGEgbG9vayBhdCBodHRwOi8vYnJvd3NlcmlmeS5vcmcvIGZvciBtb3JlIGluZm9cbipcbiovXG4vKmpzaGludCB1bnVzZWQ6IGZhbHNlICovXG4ndXNlIHN0cmljdCc7XG5cbnZhciBQSVhJID0gcmVxdWlyZSgncGl4aScpO1xudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcbnZhciBEb3VibGVSb2xsID0gcmVxdWlyZSgnLi9kb3VibGVyb2xsLmpzJyk7XG52YXIgQW5ub3RzVGltZUxpbmUgPSByZXF1aXJlKCcuL2Fubm90c3RpbWVsaW5lLmpzJyk7XG52YXIgQW5ub3RzUm9sbCA9IHJlcXVpcmUoJy4vYW5ub3Rzcm9sbC5qcycpO1xudmFyIFV0aWxzID0gcmVxdWlyZSgnLi91dGlscy5qcycpO1xuXG52YXIgZGVmYXVsdE9wdGlvbnMgPSB7XG4gICAgeEluaXQ6IDAsXG4gICAgeUluaXQ6IDAsXG4gICAgd2lkdGg6IDEwMjQsXG4gICAgaGVpZ2h0OiA3Njhcbn07XG5cbmZ1bmN0aW9uIEFubm90c1ZpelZpZXcob3B0aW9ucyl7XG4gICAgdmFyIF90aGlzID0gdGhpcztcbiAgICB2YXIgb3B0cyA9IF8ob3B0aW9ucykuZGVmYXVsdHMoZGVmYXVsdE9wdGlvbnMpLnZhbHVlKCk7XG5cbiAgICB0aGlzLmNvbnRhaW5lciA9IG5ldyBQSVhJLkRpc3BsYXlPYmplY3RDb250YWluZXIoKTtcbiAgICB0aGlzLmNvbnRhaW5lci54ID0gb3B0cy54SW5pdDtcbiAgICB0aGlzLmNvbnRhaW5lci55ID0gb3B0cy55SW5pdDtcbiAgICB0aGlzLndpZHRoID0gb3B0cy53aWR0aDtcbiAgICB0aGlzLmhlaWdodD0gb3B0cy5oZWlnaHQ7XG4gICAgdGhpcy50aW1lQmVnaW4gPSBvcHRzLnRpbWVCZWdpbjtcbiAgICB0aGlzLnRpbWVFbmQgPSBvcHRzLnRpbWVFbmQ7XG5cbiAgICB0aGlzLmFubm90Q2F0ZWdvcmllcyA9IFtdO1xuICAgIHRoaXMuYW5ub3RDYXRlZ29yaWVzRGF5MiA9IFtdO1xuICAgIHRoaXMuYW5ub3RDYXRlZ29yaWVzRGF5MSA9IFtdO1xuXG4gICAgdmFyIHdzUGlhbm9yb2xsID0gb3B0cy53c1BpYW5vcm9sbDtcbiAgICB2YXIgd3NBbm5vdCA9IG9wdHMud3NBbm5vdDtcbiAgICB2YXIgc3RhZ2VWaWV3ID0gb3B0cy5zdGFnZVZpZXc7XG4gICAgdmFyIGNoYW5uZWwgPSBvcHRzLmNoYW5uZWw7XG4gICAgdmFyIHNlcnZlclVybCA9IG9wdHMuc2VydmVyVXJsO1xuXG4gICAgdmFyIGV2ZW50Q29kZSA9IG9wdHMuZXZlbnRDb2RlO1xuICAgIHZhciBldmVudENvZGVTZXNzaW9uRGF5MiA9IG9wdHMuZXZlbnRDb2RlU2Vzc2lvbkRheTI7XG4gICAgdmFyIGV2ZW50Q29kZVNlc3Npb25EYXkxID0gb3B0cy5ldmVudENvZGVTZXNzaW9uRGF5MTtcbiAgICB2YXIgbG9nZ2VyID0gb3B0cy5sb2dnZXI7XG5cbiAgICBVdGlscy5nZXRBbm5vdENhdGVnb3JpZXMob3B0cy51cmxDYXRlZ29yaWVzICsgZXZlbnRDb2RlLCB0aGlzLmFubm90Q2F0ZWdvcmllcyk7XG4gICAgVXRpbHMuZ2V0QW5ub3RDYXRlZ29yaWVzKG9wdHMudXJsQ2F0ZWdvcmllcyArIGV2ZW50Q29kZVNlc3Npb25EYXkyLCB0aGlzLmFubm90Q2F0ZWdvcmllc0RheTIpO1xuICAgIFV0aWxzLmdldEFubm90Q2F0ZWdvcmllcyhvcHRzLnVybENhdGVnb3JpZXMgKyBldmVudENvZGVTZXNzaW9uRGF5MSwgdGhpcy5hbm5vdENhdGVnb3JpZXNEYXkxKTtcblxuICAgIHN0YWdlVmlldy5yZWdpc3RlckNvbXBvbmVudCh0aGlzKTtcblxuICAgIHZhciBjdXJyZW50RGF5ID0gbmV3IEFubm90c1RpbWVMaW5lLkFubm90c1RpbWVMaW5lKHtcbiAgICAgICAgc3RhZ2VWaWV3IDogc3RhZ2VWaWV3LFxuICAgICAgICBsb2dnZXI6IGxvZ2dlcixcbiAgICAgICAgd3M6IHdzQW5ub3QsXG4gICAgICAgIHhJbml0OiAwLFxuICAgICAgICB5SW5pdDogMCxcbiAgICAgICAgd2lkdGg6IDEwMjQgLSAyMDAgLSAyMDAsXG4gICAgICAgIGhlaWdodDogNzY4LTIwMCxcbiAgICAgICAgdGltZUJlZ2luOiB0aGlzLnRpbWVCZWdpbixcbiAgICAgICAgdGltZUVuZDogdGhpcy50aW1lRW5kLFxuICAgICAgICBpbnRlcnZhbFdpZHRoOiA2LFxuICAgICAgICBpbnRlcnZhbEhlaWdodDogMTAsXG4gICAgICAgIG1heENlbGxIZWlnaHQ6IDcwLFxuICAgICAgICByYWRpdXM6IDIwMCxcbiAgICAgICAgZXZlbnRDb2RlOiBldmVudENvZGUsXG4gICAgICAgIGNoYW5uZWw6IGNoYW5uZWwsXG4gICAgICAgIHNlcnZlclVybDogc2VydmVyVXJsLFxuICAgICAgICBhbm5vdENhdGVnb3JpZXM6IHRoaXMuYW5ub3RDYXRlZ29yaWVzXG4gICAgfSk7XG5cbiAgICAvL0FyY2hpdmUgZGF5IDJcbiAgICB2YXIgdGltZUxpbmVEYXkyID0gbmV3IEFubm90c1RpbWVMaW5lLkFubm90c1RpbWVMaW5lKHtcbiAgICAgICAgc3RhZ2VWaWV3IDogc3RhZ2VWaWV3LFxuICAgICAgICB4SW5pdDogMCxcbiAgICAgICAgeUluaXQ6IDAsXG4gICAgICAgIHdpZHRoOiAxMDI0IC0gMjAwIC0gMjAwLFxuICAgICAgICBoZWlnaHQ6IDc2OC0yMDAsXG4gICAgICAgIHRpbWVCZWdpbjogRGF0ZS5wYXJzZSgnMjAxNS0wNC0xNlQxNDowMDowMCswMjowMCcpLFxuICAgICAgICB0aW1lRW5kOiBEYXRlLnBhcnNlKCcyMDE1LTA0LTE2VDIxOjAwOjAwKzAyOjAwJyksXG4gICAgICAgIGNpcmNsZVg6IGN1cnJlbnREYXkuY2lyY2xlWCxcbiAgICAgICAgY2lyY2xlWTogY3VycmVudERheS5jaXJjbGVZLFxuICAgICAgICBpbnRlcnZhbFdpZHRoOiAoY3VycmVudERheS5yYWRpdXMqMi8zKSogY3VycmVudERheS5pbnRlcnZhbFdpZHRoIC8gY3VycmVudERheS5yYWRpdXMsXG4gICAgICAgIGludGVydmFsSGVpZ2h0OiAoY3VycmVudERheS5pbnRlcnZhbEhlaWdodCAqIChjdXJyZW50RGF5LnJhZGl1cyAtIGN1cnJlbnREYXkucmFkaXVzKjIvMykpLyBjdXJyZW50RGF5Lm1heENlbGxIZWlnaHQsXG4gICAgICAgIG1heENlbGxIZWlnaHQ6IChjdXJyZW50RGF5LnJhZGl1cyAtIGN1cnJlbnREYXkucmFkaXVzKjIvMykvMixcbiAgICAgICAgcmFkaXVzOiBjdXJyZW50RGF5LnJhZGl1cyoyLzMsXG4gICAgICAgIGFubm90Q2F0ZWdvcmllczogdGhpcy5hbm5vdENhdGVnb3JpZXNEYXkyLFxuICAgICAgICBldmVudENvZGU6IGV2ZW50Q29kZVNlc3Npb25EYXkyLFxuICAgICAgICBjaGFubmVsOiBjaGFubmVsLFxuICAgICAgICBzZXJ2ZXJVcmw6IHNlcnZlclVybCxcbiAgICAgICAgc2hvd0Nsb2NrR3JhcGhpY3M6ZmFsc2UsXG4gICAgICAgIGFyY2hpdmU6IHRydWVcbiAgICB9KTtcblxuICAgIC8vQXJjaGl2ZSBkYXkgMVxuICAgIHZhciB0aW1lTGluZURheTEgPSBuZXcgQW5ub3RzVGltZUxpbmUuQW5ub3RzVGltZUxpbmUoe1xuICAgICAgICBzdGFnZVZpZXcgOiBzdGFnZVZpZXcsXG4gICAgICAgIHhJbml0OiAwLFxuICAgICAgICB5SW5pdDogMCxcbiAgICAgICAgd2lkdGg6IDEwMjQgLSAyMDAgLSAyMDAsXG4gICAgICAgIGhlaWdodDogNzY4LTIwMCxcbiAgICAgICAgdGltZUJlZ2luOiBEYXRlLnBhcnNlKCcyMDE1LTA0LTE2VDE0OjAwOjAwKzAyOjAwJyksXG4gICAgICAgIHRpbWVFbmQ6IERhdGUucGFyc2UoJzIwMTUtMDQtMTZUMjE6MDA6MDArMDI6MDAnKSxcbiAgICAgICAgY2lyY2xlWDogY3VycmVudERheS5jaXJjbGVYLFxuICAgICAgICBjaXJjbGVZOiBjdXJyZW50RGF5LmNpcmNsZVksXG4gICAgICAgIGludGVydmFsV2lkdGg6IChjdXJyZW50RGF5LnJhZGl1cyoyLzMpKiBjdXJyZW50RGF5LmludGVydmFsV2lkdGggLyBjdXJyZW50RGF5LnJhZGl1cyxcbiAgICAgICAgaW50ZXJ2YWxIZWlnaHQ6IChjdXJyZW50RGF5LmludGVydmFsSGVpZ2h0ICogKGN1cnJlbnREYXkucmFkaXVzIC0gY3VycmVudERheS5yYWRpdXMvMykpLyBjdXJyZW50RGF5Lm1heENlbGxIZWlnaHQsXG4gICAgICAgIG1heENlbGxIZWlnaHQ6IChjdXJyZW50RGF5LnJhZGl1cyoyLzMgLSBjdXJyZW50RGF5LnJhZGl1cy8zKS80LFxuICAgICAgICByYWRpdXM6IGN1cnJlbnREYXkucmFkaXVzLzMsXG4gICAgICAgIGFubm90Q2F0ZWdvcmllczogdGhpcy5hbm5vdENhdGVnb3JpZXNEYXkxLFxuICAgICAgICBldmVudENvZGU6IGV2ZW50Q29kZVNlc3Npb25EYXkxLFxuICAgICAgICBjaGFubmVsOiBjaGFubmVsLFxuICAgICAgICBzZXJ2ZXJVcmw6IHNlcnZlclVybCxcbiAgICAgICAgc2hvd0Nsb2NrR3JhcGhpY3M6ZmFsc2UsXG4gICAgICAgIGFyY2hpdmU6IHRydWUsXG4gICAgfSk7XG5cbiAgICB2YXIgY3VycmVudFRpbWVUZXh0ID0gbmV3IFBJWEkuVGV4dCgnLS0gOiAtLSA6IC0tJywgeyBmb250OiAnMThwdCBHb3RoaWMgU3RhbmRhcmQnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG4gICAgY3VycmVudFRpbWVUZXh0LnggPSB0aW1lTGluZURheTIuY2lyY2xlWCAtIGN1cnJlbnRUaW1lVGV4dC53aWR0aC8yO1xuICAgIGN1cnJlbnRUaW1lVGV4dC55ID0gdGltZUxpbmVEYXkyLmNpcmNsZVkgLSBjdXJyZW50VGltZVRleHQuaGVpZ2h0LzI7XG4gICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY3VycmVudFRpbWVUZXh0KTtcblxuXG4gICAgdmFyIGRvdWJsZVJvbGxIID0gbmV3IERvdWJsZVJvbGwuRG91YmxlUm9sbCh7XG4gICAgICAgIHN0YWdlVmlldyA6IHN0YWdlVmlldyxcbiAgICAgICAgbG9nZ2VyOiBsb2dnZXIsXG4gICAgICAgIHdzOiB3c1BpYW5vcm9sbCxcbiAgICAgICAgeUluaXQ6ICh0aGlzLmhlaWdodCAtIDIwMCksXG4gICAgICAgIHNjZW5lSGVpZ2h0OiAyMDAsXG4gICAgICAgIHBpYW5vcm9sbHMgOiBbXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgaGVpZ2h0OiAyMDAsXG4gICAgICAgICAgICAgICAgdGltZVdpZHRoOiAxMCxcbiAgICAgICAgICAgICAgICBsaW5lSW50ZXJ2YWw6IDUwMDAsXG4gICAgICAgICAgICAgICAgbm90ZUhlaWdodDogMTBcbiAgICAgICAgICAgIH0sXG4gICAgICAgIF1cbiAgICB9KTtcblxuICAgIHZhciBkb3VibGVSb2xsViA9IG5ldyBEb3VibGVSb2xsLkRvdWJsZVJvbGwoe1xuICAgICAgICBzdGFnZVZpZXcgOiBzdGFnZVZpZXcsXG4gICAgICAgIGxvZ2dlcjogbG9nZ2VyLFxuICAgICAgICB3czogd3NQaWFub3JvbGwsXG4gICAgICAgIG9yaWVudGF0aW9uOiAndmVydGljYWwnLFxuICAgICAgICBzY2VuZUhlaWdodDogNzY4LTIwMCxcbiAgICAgICAgcGlhbm9yb2xscyA6IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICBoZWlnaHQ6IDIwMCxcbiAgICAgICAgICAgICAgICB0aW1lV2lkdGg6IDYwLFxuICAgICAgICAgICAgICAgIGxpbmVJbnRlcnZhbDogNTAwMCxcbiAgICAgICAgICAgICAgICBub3RlSGVpZ2h0OiA1LFxuICAgICAgICAgICAgfSxcbiAgICAgICAgXVxuICAgIH0pO1xuXG4gICAgdmFyIGFubm90c1JvbGwgPSBuZXcgQW5ub3RzUm9sbC5Bbm5vdHNSb2xsKHtcbiAgICAgICAgc3RhZ2VWaWV3IDogc3RhZ2VWaWV3LFxuICAgICAgICBsb2dnZXI6IGxvZ2dlcixcbiAgICAgICAgd3M6IHdzQW5ub3QsXG4gICAgICAgIHBhcmVudENvbnRhaW5lcjogZG91YmxlUm9sbFYuc3RhZ2UsXG4gICAgICAgIHhJbml0OiAxMDI0IC0gMjAwIC0gMjAwLFxuICAgICAgICB5SW5pdDogNzY4LTIwMCxcbiAgICAgICAgd2lkdGg6IDIwMCArIDIwMCxcbiAgICAgICAgaGVpZ2h0OiA3NjgtMjAwLFxuICAgICAgICB3aWR0aFJvbGw6IDIwMCxcbiAgICAgICAgZnJhbWVyYXRlOiBkb3VibGVSb2xsVi5mcmFtZXJhdGUsXG4gICAgICAgIHBpeGVsc1BlclNlY29uZDogTWF0aC5mbG9vcigxMDI0IC8gNjApLFxuICAgICAgICBhbm5vdENvbG9yczogdGhpcy5hbm5vdENhdGVnb3JpZXNcbiAgICB9KTtcblxuICAgIHZhciBsaW1pdGVycyA9IG5ldyBQSVhJLkdyYXBoaWNzKClcbiAgICAgICAgLmxpbmVTdHlsZSgxLCAweDY0NjQ2NClcbiAgICAgICAgLm1vdmVUbyhhbm5vdHNSb2xsLmNvbnRhaW5lci54LCBhbm5vdHNSb2xsLmNvbnRhaW5lci55KVxuICAgICAgICAubGluZVRvKGFubm90c1JvbGwuY29udGFpbmVyLngsIGFubm90c1JvbGwuY29udGFpbmVyLnkgLSBhbm5vdHNSb2xsLmhlaWdodClcbiAgICAgICAgLm1vdmVUbyhhbm5vdHNSb2xsLmNvbnRhaW5lci54ICsgYW5ub3RzUm9sbC53aWR0aFJvbGwsIGFubm90c1JvbGwuY29udGFpbmVyLnkpXG4gICAgICAgIC5saW5lVG8oYW5ub3RzUm9sbC5jb250YWluZXIueCArIGFubm90c1JvbGwud2lkdGhSb2xsLCBhbm5vdHNSb2xsLmNvbnRhaW5lci55IC0gYW5ub3RzUm9sbC5oZWlnaHQpXG4gICAgICAgIC5tb3ZlVG8oMCwgdGhpcy5oZWlnaHQgLSAyMDApXG4gICAgICAgIC5saW5lVG8odGhpcy53aWR0aCwgdGhpcy5oZWlnaHQgLSAyMDApXG4gICAgICAgIC5kcmF3UmVjdCgwLCAwLCB0aGlzLndpZHRoIC0xLCB0aGlzLmhlaWdodCAtMSlcbiAgICAgICAgLmJlZ2luRmlsbCgweEVDRUNFQylcbiAgICAgICAgLmRyYXdSZWN0KDEwMjQgLSAyMDAsIDAsIDIwMCwgNzY4LTIwMClcbiAgICAgICAgLmVuZEZpbGwoKTtcbiAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChsaW1pdGVycyk7XG5cbiAgICB0aGlzLmluaXQgPSBmdW5jdGlvbigpe1xuICAgIH07XG5cbiAgICB0aGlzLnVwZGF0ZVRpbWUgPSBmdW5jdGlvbigpe1xuICAgICAgICBjdXJyZW50VGltZVRleHQuc2V0VGV4dChVdGlscy5mb3JtYXRUaW1lKERhdGUubm93KCkpKTtcbiAgICB9O1xuXG4gICAgdmFyIHJlZnJlc2hUaW1lSW50ZXJ2YWw7XG5cbiAgICB0aGlzLnN0YXJ0ID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIHJlZnJlc2hUaW1lSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbigpIHtfdGhpcy51cGRhdGVUaW1lKCk7fSwgMTAwMCk7XG4gICAgfTtcblxuICAgIHRoaXMucmVmcmVzaCA9IGZ1bmN0aW9uKCkge1xuICAgIH07XG5cbiAgICB0aGlzLnN0b3AgPSBmdW5jdGlvbigpe1xuICAgIH07XG5cbiAgICByZXR1cm4gdGhpcztcblxufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBBbm5vdHNWaXpWaWV3OiBBbm5vdHNWaXpWaWV3XG59O1xuIiwiLyoqXG4qIHNjcmlwdHMvZG91Ymxlcm9sbC5qc1xuKlxuKiBUaGlzIGlzIHRoZSBzdGFydGluZyBwb2ludCBmb3IgeW91ciBhcHBsaWNhdGlvbi5cbiogVGFrZSBhIGxvb2sgYXQgaHR0cDovL2Jyb3dzZXJpZnkub3JnLyBmb3IgbW9yZSBpbmZvXG4qL1xuXG4ndXNlIHN0cmljdCc7XG5cblxudmFyIFBJWEkgPSByZXF1aXJlKCdwaXhpJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xudmFyIFBpYW5vUm9sbCA9IHJlcXVpcmUoJy4vcGlhbm9yb2xsJyk7XG52YXIgVXRpbHMgPSByZXF1aXJlKCcuL3V0aWxzLmpzJyk7XG5cbnZhciBkZWZhdWx0Q29uZmlnID0ge1xuICAgIG9yaWVudGF0aW9uOiAnaG9yaXpvbnRhbCcsXG4gICAgbG9nZ2VyOiB1bmRlZmluZWQsXG4gICAgc2NlbmVXaWR0aDogMTAyNCxcbiAgICBzaG93VGltZXI6IGZhbHNlLFxuICAgIHBpYW5vcm9sbHMgOiBbXG4gICAgICB7XG4gICAgICAgIGhlaWdodDogMzg0LFxuICAgICAgICB0aW1lV2lkdGg6IDEwLFxuICAgICAgICBsaW5lSW50ZXJ2YWw6IDUwMDAsXG4gICAgICAgIG5vdGVIZWlnaHQ6IHVuZGVmaW5lZCxcbiAgICAgICAgcmFuZ2U6IHtcbiAgICBcdFx0Ym90dG9tOiA0MCxcbiAgICBcdFx0dG9wOiA5MCxcbiAgICAgICAgfSxcbiAgICAgICAgZHluYW1pY1JhbmdlOiB0cnVlLFxuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgaGVpZ2h0OiAzODQsXG4gICAgICAgIHRpbWVXaWR0aDogNjAsXG4gICAgICAgIGxpbmVJbnRlcnZhbDogNTAwMCxcbiAgICAgICAgbm90ZUhlaWdodDogdW5kZWZpbmVkLFxuICAgICAgICByYW5nZTp7XG4gICAgXHRcdGJvdHRvbTogMCxcbiAgICBcdFx0dG9wOiAxMjgsXG4gICAgICAgIH0sXG4gICAgICAgIGR5bmFtaWNSYW5nZTogZmFsc2UsXG4gICAgICB9LFxuICAgIF0sXG4gICAgZnJhbWVyYXRlOiAyNSxcbiAgICBvZmZzZXRNdXNpYzogZmFsc2UsXG4gICAgc2NlbmVCZ0NvbG9yOiAweEZGRkZGRixcbiAgICBsaW5lQ29sb3I6IDB4NDQ0NDQ0LFxuICAgIGxpbmVGaWxsQ29sb3I6IDB4RkZGRjAwLFxuICAgIG5vdGVDb2xvcnM6IFsweEI5MDAwMCwgMHg0QkRENzEsIDB4QUY5MzFFLCAweDFDMjhCQSwgMHg1MzY5OTFdLFxuICAgIG5vdGVIZWlnaHQ6IHVuZGVmaW5lZCxcbiAgICB6ZXJvU2hpZnQ6IDAuOSxcbiAgICB0aW1lV2lkdGg6IDYwLFxuICAgIGxpbmVJbnRlcnZhbDogNTAwMCxcbi8vICAgIHdzVXJpOiB1bmRlZmluZWQsXG4vLyAgICBldmVudENvZGU6IHVuZGVmaW5lZFxuXG59O1xuXG5mdW5jdGlvbiBEb3VibGVSb2xsKG9wdGlvbnMpIHtcblxuICAgIHZhciBfdGhpcyA9IHRoaXM7XG4gICAgdmFyIG9wdHMgPSBfKG9wdGlvbnMpLmRlZmF1bHRzKGRlZmF1bHRDb25maWcpLnZhbHVlKCk7XG5cbiAgICB2YXIgb3JpZW50YXRpb24gPSBvcHRzLm9yaWVudGF0aW9uO1xuICAgIHZhciBpc0hvcml6b250YWwgPSAob3JpZW50YXRpb24gIT09ICd2ZXJ0aWNhbCcpO1xuXG4gICAgdGhpcy5sb2dnZXIgPSBvcHRzLmxvZ2dlcjtcbiAgICB0aGlzLmxpbmVDb2xvciA9IG9wdHMubGluZUNvbG9yO1xuICAgIHRoaXMubGluZUZpbGxDb2xvciA9IG9wdHMubGluZUZpbGxDb2xvcjtcbiAgICB0aGlzLmZyYW1lcmF0ZSA9IG9wdHMuZnJhbWVyYXRlO1xuICAgIHRoaXMub2Zmc2V0TXVzaWMgPSBvcHRzLm9mZnNldE11c2ljO1xuICAgIHRoaXMubm90ZUNvbG9ycyA9IG9wdHMubm90ZUNvbG9ycztcbiAgICB0aGlzLnNob3dUaW1lciA9IG9wdHMuc2hvd1RpbWVyO1xuXG4gICAgdmFyIG5vdGVIZWlnaHQgPSBvcHRzLm5vdGVIZWlnaHQ7XG4gICAgdmFyIHNjZW5lSGVpZ2h0ID0gb3B0cy5zY2VuZUhlaWdodCB8fCBfKG9wdHMucGlhbm9yb2xscykucmVkdWNlKGZ1bmN0aW9uKHMscCkgeyByZXR1cm4gcyArIHAuaGVpZ2h0OyB9LCAwKTtcbiAgICB2YXIgdGltZVdpZHRoID0gb3B0cy50aW1lV2lkdGg7XG4gICAgdmFyIGxpbmVJbnRlcnZhbCA9IG9wdHMubGluZUludGVydmFsO1xuICAgIHZhciBvZmZzZXRNdXNpYyA9IG9wdHMub2Zmc2V0TXVzaWM7XG5cbiAgICB2YXIgc2NlbmVXaWR0aCA9IG9wdHMuc2NlbmVXaWR0aDtcbiAgICB2YXIgc3RhZ2VWaWV3ID0gb3B0cy5zdGFnZVZpZXc7XG5cbiAgICBcbiAgICB2YXIgemVyb1NoaWZ0ID0gb3B0cy56ZXJvU2hpZnQ7XG5cbiAgICB2YXIgd3MgPSBvcHRzLndzO1xuXG4gICAgdmFyIGNvbG9yc1JlZyA9IHt9O1xuXG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IE1hdGguZmxvb3Ioc2NlbmVXaWR0aCp6ZXJvU2hpZnQpO1xuICAgIHRoaXMuY29udGFpbmVyLnkgPSAwO1xuXG4gICAgc3RhZ2VWaWV3LnJlZ2lzdGVyQ29tcG9uZW50KHRoaXMpO1xuXG4gICAgdmFyIHBpYW5vcm9sbExpc3QgPSBbXTtcblxuICAgIHZhciBwaWFub3JvbGxPcHRpb25zID0ge1xuICAgICAgICBwYXJlbnRDb250YWluZXI6IHRoaXMuY29udGFpbmVyLFxuICAgICAgICBvcmllbnRhdGlvbjogb3JpZW50YXRpb24sXG4gICAgICAgIHhJbml0OiAwLFxuICAgICAgICB3aWR0aDogc2NlbmVXaWR0aCxcbiAgICAgICAgbm90ZUNvbG9yczogdGhpcy5ub3RlQ29sb3JzLFxuICAgICAgICBjb2xvcnNSZWc6IGNvbG9yc1JlZyxcbiAgICAgICAgbGluZUNvbG9yOiB0aGlzLmxpbmVDb2xvcixcbiAgICAgICAgbGluZUludGVydmFsOiBsaW5lSW50ZXJ2YWwsXG4gICAgICAgIG9mZnNldE11c2ljOiBvZmZzZXRNdXNpYyxcbiAgICB9O1xuXG4gICAgdmFyIHlJbml0ID0gb3B0cy55SW5pdCB8fCAwO1xuICAgIHZhciBsaW5lc0Rvd24gPSB0cnVlO1xuICAgIF8ob3B0cy5waWFub3JvbGxzKS5mb3JFYWNoKGZ1bmN0aW9uKHByRGVmLCBpKSB7XG4gICAgICAgIHZhciBwck5vdGVIZWlnaHQgPSBub3RlSGVpZ2h0IHx8IHByRGVmLm5vdGVIZWlnaHQgfHwgcHJEZWYuaGVpZ2h0IC8gKHByRGVmLnJhbmdlLnRvcCAtIHByRGVmLnJhbmdlLmJvdHRvbSArIDEpO1xuICAgICAgICB2YXIgcHJUaW1lV2lkdGggPSBwckRlZi50aW1lV2lkdGggfHwgdGltZVdpZHRoO1xuICAgICAgICBwaWFub3JvbGxMaXN0LnB1c2gobmV3IFBpYW5vUm9sbChfKHtcbiAgICAgICAgICAgIHlJbml0OiB5SW5pdCxcbiAgICAgICAgICAgIGhlaWdodDogcHJEZWYuaGVpZ2h0LFxuICAgICAgICAgICAgbGluZXNEb3duOiBsaW5lc0Rvd24sXG4gICAgICAgICAgICBwaXhlbHNQZXJTZWNvbmQ6IE1hdGguZmxvb3Ioc2NlbmVXaWR0aCAvIHByVGltZVdpZHRoKSxcbiAgICAgICAgICAgIG5vdGVIZWlnaHQ6IHByTm90ZUhlaWdodCxcbiAgICAgICAgICAgIGxpbmVJbnRlcnZhbDogcHJEZWYubGluZUludGVydmFsLFxuICAgICAgICAgICAgcmFuZ2U6IHByRGVmLnJhbmdlLFxuICAgICAgICAgICAgZHluYW1pY1JhbmdlOiBwckRlZi5keW5hbWljUmFuZ2UsXG4gICAgICAgIH0pLmRlZmF1bHRzKHBpYW5vcm9sbE9wdGlvbnMpLnZhbHVlKCkpKTtcbiAgICAgICAgeUluaXQgKz0gcHJEZWYuaGVpZ2h0O1xuICAgICAgICBsaW5lc0Rvd24gPSAhbGluZXNEb3duO1xuXG4gICAgICAgIGlmKGk8KG9wdHMucGlhbm9yb2xscy5sZW5ndGgtMSkpIHtcbiAgICAgICAgICAgIHZhciBsaW5lR3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpXG4gICAgICAgICAgICAgICAgLmJlZ2luRmlsbChfdGhpcy5saW5lRmlsbENvbG9yKVxuICAgICAgICAgICAgICAgIC5saW5lU3R5bGUoMSwgX3RoaXMubGluZUNvbG9yKVxuICAgICAgICAgICAgICAgIC5tb3ZlVG8oTWF0aC5mbG9vcihzY2VuZVdpZHRoKnplcm9TaGlmdCksIHlJbml0KVxuICAgICAgICAgICAgICAgIC5saW5lVG8oLXNjZW5lV2lkdGggLSBNYXRoLmZsb29yKHNjZW5lV2lkdGgqemVyb1NoaWZ0KSwgeUluaXQpXG4gICAgICAgICAgICAgICAgLmVuZEZpbGwoKTtcbiAgICAgICAgICAgIF90aGlzLmNvbnRhaW5lci5hZGRDaGlsZChsaW5lR3JhcGhpY3MpO1xuICAgICAgICB9XG4gICAgfSk7XG5cbiAgICBpZiAodGhpcy5zaG93VGltZXIpe1xuXHQgICAgdmFyIGN1cnJlbnRUaW1lVGV4dCA9IG5ldyBQSVhJLlRleHQoJy0tIDogLS0gOiAtLScsIHsgZm9udDogJzQwcHQgQXJpYWwnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG5cdCAgICBjdXJyZW50VGltZVRleHQueCA9IC1jdXJyZW50VGltZVRleHQud2lkdGggLSA2MDsgXG5cdCAgICBjdXJyZW50VGltZVRleHQueSA9IHNjZW5lSGVpZ2h0IC0gY3VycmVudFRpbWVUZXh0LmhlaWdodCAtIDMwO1xuXHQgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY3VycmVudFRpbWVUZXh0KTtcblx0fVxuICAgIFxuICAgIFxuICAgIGlmKCFpc0hvcml6b250YWwpIHtcbiAgICAgICAgdGhpcy5jb250YWluZXIucm90YXRpb24gPSBNYXRoLlBJLzI7IFxuICAgICAgICB0aGlzLmNvbnRhaW5lci55ID0gc2NlbmVIZWlnaHQ7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLnggPSBzY2VuZVdpZHRoO1xuICAgIH1cblxuXG4gICAgdGhpcy5pbml0ID0gZnVuY3Rpb24oKSB7XG5cbiAgICBcdHdzLm1lc3NhZ2UoZnVuY3Rpb24oZGF0YSkge1xuICAgICAgICAgICAgX3RoaXMuYWRkTm90ZXMoZGF0YSk7XG4gICAgICAgIH0pO1xuXG4gICAgfTtcbiAgICBcbiAgICB0aGlzLnVwZGF0ZVRpbWUgPSBmdW5jdGlvbigpe1xuICAgICAgICBjdXJyZW50VGltZVRleHQuc2V0VGV4dChVdGlscy5mb3JtYXRUaW1lKERhdGUubm93KCkpKTtcbiAgICB9O1xuXG4gICAgdGhpcy5hZGROb3RlcyA9IGZ1bmN0aW9uKGRhdGEpIHtcblxuICAgICAgICBwaWFub3JvbGxMaXN0LmZvckVhY2goZnVuY3Rpb24oYykge1xuICAgICAgICAgICAgYy5hZGROb3RlUmF3KGRhdGEpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIHBpYW5vcm9sbExpc3QuZm9yRWFjaChmdW5jdGlvbihjKSB7XG4gICAgICAgICAgICBjLm1vdmUoKTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIC8vIEluaXQgcGFnZSBhbmQgaW50ZXJ2YWxzXG4gICAgdmFyIHN0YXJ0VHM7XG4gICAgdmFyIHJlZnJlc2hUaW1lSW50ZXJ2YWw7XG5cbiAgICB0aGlzLnN0YXJ0ID0gZnVuY3Rpb24oKSB7XG4gICAgXHRpZiAodGhpcy5zaG93VGltZXIpe1xuICAgIFx0XHRyZWZyZXNoVGltZUludGVydmFsID0gc2V0SW50ZXJ2YWwoZnVuY3Rpb24oKSB7X3RoaXMudXBkYXRlVGltZSgpO30sIDEwMDApO1xuICAgIFx0fVxuICAgIFx0XG4gICAgICAgIHN0YXJ0VHMgPSBEYXRlLm5vdygpO1xuICAgICAgICBwaWFub3JvbGxMaXN0LmZvckVhY2goZnVuY3Rpb24oYykge1xuICAgICAgICAgICAgYy5zdGFydCgpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5zdG9wID0gZnVuY3Rpb24oKSB7XG5cbiAgICAgICAgcGlhbm9yb2xsTGlzdC5mb3JFYWNoKGZ1bmN0aW9uKGMpIHtcbiAgICAgICAgICAgIGMuc3RvcCgpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG5cbiAgICB0aGlzLmxvZyA9IGZ1bmN0aW9uKG0pIHtcbiAgICAgICAgaWYodGhpcy5sb2dnZXIpIHtcbiAgICAgICAgICAgIHRoaXMubG9nZ2VyLmxvZyhtKTtcbiAgICAgICAgfVxuICAgIH07XG5cblxuXG4gICAgcmV0dXJuIHRoaXM7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIERvdWJsZVJvbGw6IERvdWJsZVJvbGxcbn07XG4iLCIvKipcbioganMvd3N3cmFwcGVyLmpzXG4qXG4qIHNpbXBsZSBsb2dnZXIgc2VydmljZVxuKlxuKi9cblxuLyogZ2xvYmFsIGRvY3VtZW50OiBmYWxzZSAqL1xuXG4ndXNlIHN0cmljdCc7XG5cbmZ1bmN0aW9uIEh0bWxMb2dnZXIoZG9Mb2csIGNvbnRhaW5lcikge1xuXG4gICAgdmFyIGxvZ0NvbnRhaW5lciA9IGNvbnRhaW5lcjtcbiAgICBpZih0eXBlb2YoY29udGFpbmVyKSA9PT0gJ3N0cmluZycpIHtcbiAgICAgICAgbG9nQ29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoY29udGFpbmVyKTtcbiAgICB9XG4gICAgaWYoIWRvTG9nKSB7XG4gICAgICAgIGRvY3VtZW50LmJvZHkucmVtb3ZlQ2hpbGQobG9nQ29udGFpbmVyKTtcbiAgICAgICAgbG9nQ29udGFpbmVyID0gdW5kZWZpbmVkO1xuICAgIH1cblxuXG4gICAgdGhpcy5sb2cgPSBmdW5jdGlvbihtc2cpIHtcbiAgICAgICAgaWYoZG9Mb2cgJiYgbG9nQ29udGFpbmVyKSB7XG4gICAgICAgICAgICBsb2dDb250YWluZXIuaW5uZXJIVE1MICs9IG1zZyArICdcXG4nO1xuICAgICAgICAgICAgbG9nQ29udGFpbmVyLnNjcm9sbFRvcCA9IGxvZ0NvbnRhaW5lci5zY3JvbGxIZWlnaHQ7XG4gICAgICAgIH1cbiAgICB9O1xufVxuXG5mdW5jdGlvbiBDb25zb2xlTG9nZ2VyKGRvTG9nKSB7XG5cbiAgICB0aGlzLmxvZyA9IGZ1bmN0aW9uKG1zZykge1xuICAgICAgICBpZihkb0xvZykge1xuICAgICAgICAgICAgY29uc29sZS5sb2cobXNnKTtcbiAgICAgICAgfVxuICAgIH07XG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgSHRtbExvZ2dlcjogSHRtbExvZ2dlcixcbiAgICBDb25zb2xlTG9nZ2VyOiBDb25zb2xlTG9nZ2VyXG59O1xuIiwiLyoqXG4qIGpzL3BpYW5vcm9sbC5qc1xuKlxuKiBwaWFub3JvbGwgYmFzaWMgY29tcG9uZW50XG4qXG4qL1xuXG4vL1RPRE86IGFkZCBkZWxheSB0byBhZGp1c3QgdGhlIFwiemVyb1wiLiB0aGlzIGNvdWxkIHNob3VsZCBiZSBnaXZlbiBhcyBhbiBvcHRpb25cbi8vb3IgaWYgZXhwbGljaXRlbHkgbnVsbCBhdXRvbWF0aWNhbGx5IGFkanVzdGVkIGZyb20gdGhlIG1lYW4gb2YgZGlmZmVyZW5jZVxuLy9iZXR3ZWVuIHRoZSBub3RlIHRpbWVzdGFtcCBhbmQgdGhlIGN1cnJlbnQgdGltZXN0YW1wLlxuXG4ndXNlIHN0cmljdCc7XG5cblxudmFyIFBJWEkgPSByZXF1aXJlKCdwaXhpJyk7XG52YXIgcmFuZG9tQ29sb3IgPSByZXF1aXJlKCdyYW5kb21Db2xvcicpO1xudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcblxudmFyIE5UUF9FUE9DSF9ERUxUQSA9IDIyMDg5ODg4MDA7IC8vYy5mLiBSRkMgODY4XG5cbmZ1bmN0aW9uIFBpYW5vUm9sbChvcHRpb25zKSB7XG4gICAgdmFyIF90aGlzID0gdGhpcztcbiAgICB0aGlzLmNvbnRhaW5lciA9IG5ldyBQSVhJLkRpc3BsYXlPYmplY3RDb250YWluZXIoKTtcbiAgICB0aGlzLmNvbnRhaW5lci54ID0gb3B0aW9ucy54SW5pdDtcbiAgICB0aGlzLmNvbnRhaW5lci55ID0gb3B0aW9ucy55SW5pdDtcbiAgICBvcHRpb25zLnBhcmVudENvbnRhaW5lci5hZGRDaGlsZCh0aGlzLmNvbnRhaW5lcik7XG5cbiAgICB2YXIgb3JpZW50YXRpb24gPSBvcHRpb25zLm9yaWVudGF0aW9uO1xuICAgIHZhciBpc0hvcml6b250YWwgPSAob3JpZW50YXRpb24gIT09ICd2ZXJ0aWNhbCcpO1xuXG4gICAgdGhpcy5saW5lc0Rvd24gPSBvcHRpb25zLmxpbmVzRG93bjtcbiAgICB0aGlzLmhlaWdodCA9IG9wdGlvbnMuaGVpZ2h0O1xuICAgIHRoaXMucGl4ZWxzUGVyU2Vjb25kID0gb3B0aW9ucy5waXhlbHNQZXJTZWNvbmQ7XG4gICAgdGhpcy53aWR0aCA9IG9wdGlvbnMud2lkdGg7XG4gICAgdGhpcy5ub3RlQ29sb3JzID0gb3B0aW9ucy5ub3RlQ29sb3JzO1xuICAgIHRoaXMuY29sb3JzUmVnID0gb3B0aW9ucy5jb2xvcnNSZWcgfHwge307XG4gICAgdGhpcy5saW5lQ29sb3IgPSBvcHRpb25zLmxpbmVDb2xvcjtcbiAgICB0aGlzLmxpbmVJbnRlcnZhbCA9IG9wdGlvbnMubGluZUludGVydmFsO1xuICAgIHRoaXMub2Zmc2V0TXVzaWMgPSBvcHRpb25zLm9mZnNldE11c2ljIHx8IGZhbHNlO1xuICAgIHRoaXMubm90ZUhlaWdodCA9IG9wdGlvbnMubm90ZUhlaWdodDtcbiAgICB0aGlzLm5vdGVEaWN0ID0ge307XG4gICAgdGhpcy5zdGFydFRzID0gb3B0aW9ucy5zdGFydFRzIHx8IERhdGUubm93KCk7XG4gICAgdGhpcy5keW5hbWljUmFuZ2UgPSBvcHRpb25zLmR5bmFtaWNSYW5nZTtcbiAgICB0aGlzLmluaXRSYW5nZSA9IG9wdGlvbnMucmFuZ2U7XG4gICAgdGhpcy5yYW5nZSA9IG9wdGlvbnMucmFuZ2U7XG5cblxuICAgIHZhciBzdGFydGVkID0gZmFsc2U7XG5cbiAgICB2YXIgaXNIaWRkZW4gPSBmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICAvLyBUT0RPOiB0aGUgb3JpZ2luIHBvaW50IGlzIGFuIGFwcHJveGltYXRpb24uIFNob3VsZCByZWZpbmUgdGhpc1xuICAgICAgICB2YXIgZ2xvYmFsUG9zID0gY2hpbGQudG9HbG9iYWwobmV3IFBJWEkuUG9pbnQoMCwwKSk7XG4gICAgICAgIHJldHVybiAoKGdsb2JhbFBvcy54ICsgY2hpbGQud2lkdGgpIDwgMCkgfHwgKChnbG9iYWxQb3MueSArIGNoaWxkLmhlaWdodCkgPCAwKSA7XG4gICAgfTtcblxuICAgIC8vVE9ETzogSSBkbyBub3QgbGlrZSB0aGUgXCJyZWdDb2xvclwiIG9iamVjdC4gVGhpcyBzaG91bGQgbm90IGJlIGdsb2JhbCwgYnV0IGxvY2FsXG4gICAgdGhpcy5nZXRDb2xvciA9IGZ1bmN0aW9uKGNhbmFsKSB7XG4gICAgICAgIHZhciBjb2xvciA9IHRoaXMuY29sb3JzUmVnW2NhbmFsXTtcbiAgICAgICAgaWYodHlwZW9mKGNvbG9yKSA9PT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgICAgICAgIHZhciBjb2xvcnNSZWdTaXplID0gT2JqZWN0LmtleXModGhpcy5jb2xvcnNSZWcpLmxlbmd0aDtcbiAgICAgICAgICAgIGlmKGNvbG9yc1JlZ1NpemUgPCB0aGlzLm5vdGVDb2xvcnMubGVuZ3RoKSB7XG4gICAgICAgICAgICAgICAgY29sb3IgPSB0aGlzLmNvbG9yc1JlZ1tjYW5hbF0gPSB0aGlzLm5vdGVDb2xvcnNbY29sb3JzUmVnU2l6ZV07XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBjb2xvciA9IHRoaXMuY29sb3JzUmVnW2NhbmFsXSA9IHBhcnNlSW50KHJhbmRvbUNvbG9yKHsgbHVtaW5vc2l0eTogJ2xpZ2h0JywgaHVlOiAncmFuZG9tJywgZm9ybWF0OidoZXgnfSkucmVwbGFjZSgvXiMvLCAnJyksIDE2KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gY29sb3I7XG4gICAgfTtcblxuICAgIHRoaXMuZ2V0Tm90ZVJlY3QgPSBmdW5jdGlvbihub3RlLCB4LCB5LCBjb2xvciwgYWxwaGEsIHdpZHRoLCBoZWlnaHQpIHtcbiAgICAgICAgdmFyIGdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKTtcbiAgICAgICAgZ3JhcGhpY3Mubm90ZSA9IG5vdGU7XG4gICAgICAgIGdyYXBoaWNzLmJlZ2luRmlsbChjb2xvciwgYWxwaGEpO1xuICAgICAgICBncmFwaGljcy5kcmF3UmVjdCgwLCAwLCB3aWR0aCwgaGVpZ2h0KTtcbiAgICAgICAgZ3JhcGhpY3MuZW5kRmlsbCgpO1xuICAgICAgICBncmFwaGljcy54ID0geDtcbiAgICAgICAgZ3JhcGhpY3MueSA9IHk7XG4gICAgICAgIGdyYXBoaWNzLndpZHRoID0gd2lkdGg7XG4gICAgICAgIGdyYXBoaWNzLmhlaWdodCA9IGhlaWdodDtcbiAgICAgICAgcmV0dXJuIGdyYXBoaWNzO1xuICAgIH07XG5cbiAgICB0aGlzLmFkZE5vdGVSYXcgPSBmdW5jdGlvbihkYXRhKSB7XG4gICAgICAgIHZhciBub3RlID0gZGF0YS5jb250ZW50WzNdO1xuICAgICAgICB2YXIgdmVsb2NpdHkgPSBkYXRhLmNvbnRlbnRbNF07XG4gICAgICAgIC8vdmFyIHRzID0gKGRhdGEuY29udGVudFswXSAtIE5UUF9FUE9DSF9ERUxUQSkqMTAwMDtcbiAgICAgICAgdmFyIHRzID0gRGF0ZS5wYXJzZShkYXRhLnRzX3NlcnZlcik7XG4gICAgICAgIGNvbnNvbGUubG9nKHRzLCBkYXRhLmNvbnRlbnRbMF0pO1xuICAgICAgICB2YXIgY2hhbm5lbCA9IGRhdGEuY29udGVudFsyXTtcbiAgICAgICAgdmFyIHNlc3Npb25UcyA9IGRhdGEuY29udGVudFsxXTtcblxuICAgICAgICBpZih2ZWxvY2l0eSAhPT0gMCAmJiB0eXBlb2YodGhpcy5ub3RlRGljdFtjaGFubmVsXSkgIT09ICd1bmRlZmluZWQnICYmIHR5cGVvZih0aGlzLm5vdGVEaWN0W2NoYW5uZWxdW25vdGVdICE9PSAndW5kZWZpbmVkJykpIHtcbiAgICAgICAgICAgIHRoaXMuYWRkTm90ZShub3RlLCB0cywgc2Vzc2lvblRzLCAwLCBjaGFubmVsLCAwKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuYWRkTm90ZShub3RlLCB0cywgc2Vzc2lvblRzLCB2ZWxvY2l0eSwgY2hhbm5lbCwgMCk7XG4gICAgfTtcblxuICAgIHRoaXMuYWRkTm90ZSA9IGZ1bmN0aW9uKG5vdGUsIHN0YXJ0VGltZSwgc2Vzc2lvblRzLCB2ZWxvY2l0eSwgY2hhbm5lbCwgZHVyYXRpb24pIHtcblxuICAgICAgICB2YXIgdHMgPSBzdGFydFRpbWU7XG4gICAgICAgIGlmKHRoaXMub2Zmc2V0TXVzaWMpIHtcbiAgICAgICAgICAgIHRzID0gdGhpcy5zdGFydFRzICsgc2Vzc2lvblRzO1xuICAgICAgICB9XG5cbiAgICAgICAgdmFyIG5vdGVEdXJhdGlvbiA9IGR1cmF0aW9uO1xuICAgICAgICB2YXIgbm90ZVZlbG9jaXR5ID0gdmVsb2NpdHk7XG4gICAgICAgIHZhciBncmFwaGljcztcbiAgICAgICAgaWYoIWR1cmF0aW9uKSB7XG4gICAgICAgICAgICBpZih0eXBlb2YgdGhpcy5ub3RlRGljdFtjaGFubmVsXT09PSd1bmRlZmluZWQnKXtcbiAgICAgICAgICAgICAgICB0aGlzLm5vdGVEaWN0W2NoYW5uZWxdID0ge307XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBpZih2ZWxvY2l0eT09PTApIHtcbiAgICAgICAgICAgICAgICBpZih0eXBlb2YgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXSAhPT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgICAgICAgICAgICAgICAgdmFyIG5vdGVEZWYgPSB0aGlzLm5vdGVEaWN0W2NoYW5uZWxdW25vdGVdO1xuICAgICAgICAgICAgICAgICAgICBkZWxldGUgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXTtcbiAgICAgICAgICAgICAgICAgICAgbm90ZUR1cmF0aW9uID0gc2Vzc2lvblRzIC0gbm90ZURlZi5zZXNzaW9uVHM7XG4gICAgICAgICAgICAgICAgICAgIGdyYXBoaWNzID0gbm90ZURlZi5ncmFwaGljcztcbiAgICAgICAgICAgICAgICAgICAgbm90ZVZlbG9jaXR5ID0gbm90ZURlZi52ZWxvY2l0eTtcbiAgICAgICAgICAgICAgICAgICAgdHMgPSBub3RlRGVmLnRzO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgIG5vdGVEdXJhdGlvbiA9IE1hdGguYWJzKERhdGUubm93KCkgLSB0cyk7XG4gICAgICAgICAgICAgICAgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXSA9IHsgdHM6IHRzLCB2ZWxvY2l0eTogdmVsb2NpdHksIHNlc3Npb25Uczogc2Vzc2lvblRzfTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG5cbiAgICAgICAgaWYoIXRoaXMub2Zmc2V0TXVzaWMgfHwgdmVsb2NpdHk9PT0wKSB7XG5cbiAgICAgICAgICAgIHZhciB3aWR0aCA9IG5vdGVEdXJhdGlvbiAqIHRoaXMucGl4ZWxzUGVyU2Vjb25kIC8gMTAwMDtcbiAgICAgICAgICAgIGlmKCFncmFwaGljcykge1xuICAgICAgICAgICAgICAgIHZhciB4ID0gKHRzLXRoaXMuc3RhcnRUcykgKiB0aGlzLnBpeGVsc1BlclNlY29uZCAvIDEwMDA7XG4gICAgICAgICAgICAgICAgaWYoKHgrd2lkdGgpIDwgIChNYXRoLmFicyh0aGlzLmNvbnRhaW5lci54KSAtIHRoaXMud2lkdGgpKSB7XG4gICAgICAgICAgICAgICAgICAgIC8vIG5vdCB2aXNpYmxlLiBkbyBub3RoaW5nXG4gICAgICAgICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICBcdGlmICh0aGlzLmR5bmFtaWNSYW5nZSAmJiAodGhpcy5yYW5nZS5ib3R0b20gPiBub3RlIHx8IG5vdGUgPiB0aGlzLnJhbmdlLnRvcCkpe1xuICAgICAgICAgICAgXHRcdHZhciBuZXdTY2FsZSA9IHt9O1xuICAgICAgICAgICAgXHRcdG5ld1NjYWxlWydib3R0b20nXSA9IE1hdGgubWluKG5vdGUsIHRoaXMucmFuZ2UuYm90dG9tKTtcbiAgICAgICAgICAgIFx0XHRuZXdTY2FsZVsndG9wJ10gPSBNYXRoLm1heChub3RlLCB0aGlzLnJhbmdlLnRvcCk7XG4gICAgICAgICAgICBcdFx0dGhpcy5yZXNjYWxlU2NlbmUobmV3U2NhbGUpO1xuICAgICAgICAgICAgXHR9XG4gICAgICAgICAgICBcdHZhciB5ID0gTWF0aC5mbG9vcigoKHRoaXMucmFuZ2UudG9wLXRoaXMucmFuZ2UuYm90dG9tKS0obm90ZS10aGlzLnJhbmdlLmJvdHRvbSkrMC41KSAqICh0aGlzLm5vdGVIZWlnaHQpIC0gKHRoaXMubm90ZUhlaWdodC8yKSk7XG4gICAgICAgICAgICAgICAgdmFyIGNvbG9yID0gdGhpcy5nZXRDb2xvcihjaGFubmVsKTtcblxuICAgICAgICAgICAgICAgIHZhciBhbHBoYSA9IChub3RlVmVsb2NpdHkgLyAxMjgpO1xuXG4gICAgICAgICAgICAgICAgZ3JhcGhpY3MgPSB0aGlzLmdldE5vdGVSZWN0KG5vdGUsIHgsIHksIGNvbG9yLCBhbHBoYSwgd2lkdGgsIHRoaXMubm90ZUhlaWdodCk7XG4gICAgICAgICAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoZ3JhcGhpY3MpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZ3JhcGhpY3Mud2lkdGggPSB3aWR0aDtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgaWYoIWR1cmF0aW9uICYmIHZlbG9jaXR5KSB7XG4gICAgICAgICAgICAgICAgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXS5ncmFwaGljcyA9IGdyYXBoaWNzO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgfTtcblxuICAgIC8vcmVzY2FsZSBzY2VuZSBpbiBjYXNlIGEgbm90ZSBvdXQgb2YgcmFuZ2UgaXMgYWRkZWQgb3IgYSBub3RlIG91dCBvZiB0aGUgcmFuZ2UgaXMgcmVtb3ZlZFxuICAgIHRoaXMucmVzY2FsZVNjZW5lID0gZnVuY3Rpb24obmV3U2NhbGUpe1xuICAgIFx0dmFyIF90aGlzID0gdGhpcztcbiAgICBcdHZhciBjaGlsZHJlblRvVXBkYXRlID0gW107XG4gICAgXHR2YXIgdG9wID0gdGhpcy5pbml0UmFuZ2UudG9wO1xuICAgIFx0dmFyIGJvdHRvbSA9IHRoaXMuaW5pdFJhbmdlLmJvdHRvbTtcbiAgICBcdF8oX3RoaXMuY29udGFpbmVyLmNoaWxkcmVuKS5mb3JFYWNoKGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgXHRcdGlmICh0eXBlb2YoY2hpbGQpICE9PSAndW5kZWZpbmVkJyAmJiBjaGlsZC5ub3RlICYmICFpc0hpZGRlbihjaGlsZCkpe1xuICAgIFx0XHRcdHRvcCA9IE1hdGgubWF4KGNoaWxkLm5vdGUsIHRvcCk7XG4gICAgXHRcdFx0Ym90dG9tID0gTWF0aC5taW4oY2hpbGQubm90ZSwgYm90dG9tKTtcbiAgICBcdFx0XHRyZXR1cm4gY2hpbGRyZW5Ub1VwZGF0ZS5wdXNoKGNoaWxkKTtcbiAgICBcdFx0fVxuICAgICAgICB9KTtcbiAgICBcdGlmIChuZXdTY2FsZSl7XG4gICAgXHRcdHRoaXMucmFuZ2UgPSBuZXdTY2FsZTtcbiAgICBcdH1lbHNle1xuICAgIFx0XHR0aGlzLnJhbmdlLnRvcCA9IHRvcDtcbiAgICBcdFx0dGhpcy5yYW5nZS5ib3R0b20gPSBib3R0b207XG4gICAgXHR9XG4gICAgXHR0aGlzLm5vdGVIZWlnaHQgPSB0aGlzLmhlaWdodCAvICh0aGlzLnJhbmdlLnRvcCAtIHRoaXMucmFuZ2UuYm90dG9tICsgMSk7XG4gICAgXHRjaGlsZHJlblRvVXBkYXRlLmZvckVhY2goZnVuY3Rpb24oY2hpbGQpIHtcblx0XHRcdGNoaWxkLnkgPSBNYXRoLmZsb29yKCgoX3RoaXMucmFuZ2UudG9wLV90aGlzLnJhbmdlLmJvdHRvbSktKGNoaWxkLm5vdGUtX3RoaXMucmFuZ2UuYm90dG9tKSswLjUpICogKF90aGlzLm5vdGVIZWlnaHQpIC0gKF90aGlzLm5vdGVIZWlnaHQvMikpO1xuXHRcdFx0Y2hpbGQuaGVpZ2h0ID0gX3RoaXMubm90ZUhlaWdodDtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIHRoaXMuYWRkTGluZSA9IGZ1bmN0aW9uKHRzKXtcblxuICAgICAgICBpZih0eXBlb2YodHMpID09PSAndW5kZWZpbmVkJykge1xuICAgICAgICAgICAgdHMgPSBuZXcgRGF0ZSgpO1xuICAgICAgICB9XG4gICAgICAgIHZhciB4ID0gLXRoaXMuY29udGFpbmVyLng7XG4gICAgICAgIHZhciB5ID0gdGhpcy5saW5lc0Rvd24gPyB0aGlzLmhlaWdodCAtIDIwIDogMDtcblxuICAgICAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpXG4gICAgICAgICAgICAuYmVnaW5GaWxsKDB4RkZGRjAwKVxuICAgICAgICAgICAgLmxpbmVTdHlsZSgxLCB0aGlzLmxpbmVDb2xvcilcbiAgICAgICAgICAgIC5tb3ZlVG8oMCwgMClcbiAgICAgICAgICAgIC5saW5lVG8oMCwgMjApXG4gICAgICAgICAgICAuZW5kRmlsbCgpO1xuICAgICAgICBncmFwaGljcy54ID0geDtcbiAgICAgICAgZ3JhcGhpY3MueSA9IHk7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGdyYXBoaWNzKTtcbiAgICAgICAgLy8gQWRkIHRleHRcbiAgICAgICAgLy92YXIgdG90YWxTZWMgPSBsaW5lTmIgKiB0aGlzLmxpbmVJbnRlcnZhbCAvIDEwMDA7XG4gICAgICAgIHZhciBob3VycyA9IHRzLmdldEhvdXJzKCk7XG4gICAgICAgIHZhciBtaW51dGVzID10cy5nZXRNaW51dGVzKCk7XG4gICAgICAgIHZhciBzZWNvbmRzID0gdHMuZ2V0U2Vjb25kcygpO1xuICAgICAgICB2YXIgdGltZVN0ciA9IChob3VycyA8IDEwID8gJzAnICsgaG91cnMgOiBob3VycykgKyAnOicgKyAobWludXRlcyA8IDEwID8gJzAnICsgbWludXRlcyA6IG1pbnV0ZXMpICsgJzonICsgKHNlY29uZHMgIDwgMTAgPyAnMCcgKyBzZWNvbmRzIDogc2Vjb25kcyk7XG5cbiAgICAgICAgdmFyIGZvbnRPYmogPSB7IGZvbnQ6ICcxMHB0IEFyaWFsJywgZmlsbDogJyM0NDQ0NDQnIH07XG4gICAgICAgIHZhciB0ID0gbmV3IFBJWEkuVGV4dCh0aW1lU3RyLCBmb250T2JqKTtcbiAgICAgICAgaWYoaXNIb3Jpem9udGFsKSB7XG4gICAgICAgICAgICB0LnggPSB4ICsgMjtcbiAgICAgICAgICAgIHQueSA9IHRoaXMubGluZXNEb3duID8gdGhpcy5oZWlnaHQgLSAxNSA6IDI7XG4gICAgICAgIH1cbiAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICB0LnJvdGF0aW9uID0gLU1hdGguUEkvMjtcbiAgICAgICAgICAgIHQueCA9IHggO1xuICAgICAgICAgICAgdC55ID0gdGhpcy5saW5lc0Rvd24gPyB0aGlzLmhlaWdodCAtIDIgOiB0LndpZHRoICsgMjtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZCh0KTtcbiAgICB9O1xuXG4gICAgdGhpcy5tb3ZlVG8gPSBmdW5jdGlvbihkaWZmVGltZSl7XG4gICAgICAgIHZhciBvbGRYID0gdGhpcy5jb250YWluZXIueDtcbiAgICAgICAgdGhpcy5jb250YWluZXIueCA9IE1hdGguZmxvb3IoZGlmZlRpbWUqdGhpcy5waXhlbHNQZXJTZWNvbmQpO1xuICAgICAgICB2YXIgZGVsdGFYID0gTWF0aC5hYnMob2xkWC10aGlzLmNvbnRhaW5lci54KTtcbiAgICAgICAgXy5mb3JPd24odGhpcy5ub3RlRGljdCwgZnVuY3Rpb24oY2hhbm5lbERpY3QpIHtcbiAgICAgICAgICAgIF8uZm9yT3duKGNoYW5uZWxEaWN0LCBmdW5jdGlvbihub3RlRGVmKSB7XG4gICAgICAgICAgICAgICAgaWYobm90ZURlZi5ncmFwaGljcykge1xuICAgICAgICAgICAgICAgICAgICBub3RlRGVmLmdyYXBoaWNzLndpZHRoID0gbm90ZURlZi5ncmFwaGljcy53aWR0aCArIGRlbHRhWDtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIHRoaXMubW92ZSA9IGZ1bmN0aW9uKCkge1xuICAgICAgICB2YXIgZGlmZiA9ICh0aGlzLnN0YXJ0VHMgLSBEYXRlLm5vdygpKS8xMDAwO1xuICAgICAgICB0aGlzLm1vdmVUbyhkaWZmKTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZW1vdmVQYXNzZWRPYmpldHMgPSBmdW5jdGlvbigpe1xuICAgICAgICB2YXIgY2hpbGRyZW5Ub1JlbW92ZSA9IFtdO1xuICAgICAgICB2YXIgcmVzY2FsZSA9IGZhbHNlO1xuICAgICAgICBfKF90aGlzLmNvbnRhaW5lci5jaGlsZHJlbikuZm9yRWFjaChmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICAgICAgcmV0dXJuIHR5cGVvZihjaGlsZCkgPT09ICd1bmRlZmluZWQnIHx8XG4gICAgICAgICAgICAgICAgKGlzSGlkZGVuKGNoaWxkKSAmJiBjaGlsZHJlblRvUmVtb3ZlLnB1c2goY2hpbGQpKTtcbiAgICAgICAgfSk7XG4gICAgICAgIGNoaWxkcmVuVG9SZW1vdmUuZm9yRWFjaChmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICBcdGlmIChfdGhpcy5keW5hbWljUmFuZ2UgJiYgKF90aGlzLnJhbmdlLmJvdHRvbSA9PT0gY2hpbGQubm90ZSB8fCBjaGlsZC5ub3RlID09PSBfdGhpcy5yYW5nZS50b3ApKXtcbiAgICAgICAgICAgICAgICByZXNjYWxlID0gdHJ1ZTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIF90aGlzLmNvbnRhaW5lci5yZW1vdmVDaGlsZChjaGlsZCk7XG4gICAgICAgIH0pO1xuICAgICAgICAvL2V4dGVybmFsaXplIHRoaXMgdGVzdCB0byBhdm9pZCByZXBlYXRlZCBjYWxsIHRvIHRoZSBmdW5jdGlvbiByZXNjYWxlU2NlbmUgaW4gdGhlIHByZXZpb3VzIGxvb3BcbiAgICAgICAgaWYgKHJlc2NhbGUpe1xuICAgICAgICAgICAgX3RoaXMucmVzY2FsZVNjZW5lKCk7XG4gICAgICAgIH1cbiAgICB9O1xuXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBpZighc3RhcnRlZCkge1xuICAgICAgICAgICAgdGhpcy5zdGFydFRzID0gRGF0ZS5ub3coKTtcbiAgICAgICAgICAgIHRoaXMuYWRkTGluZSgpO1xuICAgICAgICAgICAgc3RhcnRlZCA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy52ZXJ0aWNhbExpbmVzSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbigpIHsgX3RoaXMuYWRkTGluZSgpOyB9LCB0aGlzLmxpbmVJbnRlcnZhbCk7XG4gICAgICAgIHRoaXMuY2xlYW5JbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uICgpIHsgX3RoaXMucmVtb3ZlUGFzc2VkT2JqZXRzKCk7IH0sIDEwMDAgKiB0aGlzLndpZHRoIC8gdGhpcy5waXhlbHNQZXJTZWNvbmQgKTtcbiAgICB9O1xuXG4gICAgdGhpcy5zdG9wID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIGNsZWFySW50ZXJ2YWwodGhpcy52ZXJ0aWNhbExpbmVzSW50ZXJ2YWwpO1xuICAgICAgICBjbGVhckludGVydmFsKHRoaXMuY2xlYW5JbnRlcnZhbCk7XG4gICAgfTtcblxufVxuXG5tb2R1bGUuZXhwb3J0cyA9IFBpYW5vUm9sbDtcbiIsIi8qKlxuKiBzY3JpcHRzL3N0YWdldmlldy5qc1xuKlxuKiBUaGlzIGlzIHRoZSBzdGFydGluZyBwb2ludCBmb3IgeW91ciBhcHBsaWNhdGlvbi5cbiogVGFrZSBhIGxvb2sgYXQgaHR0cDovL2Jyb3dzZXJpZnkub3JnLyBmb3IgbW9yZSBpbmZvXG4qL1xuXG4vKiBnbG9iYWwgZG9jdW1lbnQ6IGZhbHNlICovXG5cbid1c2Ugc3RyaWN0JztcblxuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG5cbnZhciBkZWZhdWx0Q29uZmlnID0ge1xuICAgIGV4dGVybmFsUmVmcmVzaDogZmFsc2UsXG4gICAgbG9nZ2VyOiB1bmRlZmluZWQsXG4gICAgc2NlbmVXaWR0aDogMTAyNCxcbiAgICBzY2VuZUhlaWdodDogNzY4LFxuICAgIGZyYW1lcmF0ZTogMjUsXG4gICAgc2NlbmVCZ0NvbG9yOiAweEZGRkZGRixcbiAgICBjYW52YXNDb250YWluZXI6ICdjYW52YXNDb250YWluZXInLFxufTtcblxuZnVuY3Rpb24gU3RhZ2VWaWV3KG9wdGlvbnMpIHtcblxuICAgIHZhciBfdGhpcyA9IHRoaXM7XG4gICAgdmFyIG9wdHMgPSBfKG9wdGlvbnMpLmRlZmF1bHRzKGRlZmF1bHRDb25maWcpLnZhbHVlKCk7XG5cbiAgICB2YXIgZXh0ZXJuYWxSZWZyZXNoID0gb3B0cy5leHRlcm5hbFJlZnJlc2g7XG5cbiAgICB0aGlzLmxvZ2dlciA9IG9wdHMubG9nZ2VyO1xuICAgIHRoaXMuZnJhbWVyYXRlID0gb3B0cy5mcmFtZXJhdGU7XG4gICAgdmFyIHNjZW5lQmdDb2xvciA9IG9wdHMuc2NlbmVCZ0NvbG9yO1xuICAgIHZhciBzY2VuZVdpZHRoID0gb3B0cy5zY2VuZVdpZHRoO1xuICAgIHZhciBzY2VuZUhlaWdodCA9IG9wdHMuc2NlbmVIZWlnaHQ7XG4gICAgdmFyIGNhbnZhc0NvbnRhaW5lciA9IG9wdHMuY2FudmFzQ29udGFpbmVyO1xuICAgIHZhciB0aW1lQ29udGFpbmVyID0gW107XG4gICAgdmFyIGNvbXBvbmVudHMgPSBbXTtcblxuICAgIC8vY3JlYXRlIGFuIG5ldyBpbnN0YW5jZSBvZiBhIHBpeGkgc3RhZ2VcbiAgICB0aGlzLnN0YWdlID0gbmV3IFBJWEkuU3RhZ2Uoc2NlbmVCZ0NvbG9yKTtcbiAgICAvL2NyZWF0ZSBhIHJlbmRlcmVyIGluc3RhbmNlLlxuICAgIHZhciByZW5kZXJlciA9IFBJWEkuYXV0b0RldGVjdFJlbmRlcmVyKHNjZW5lV2lkdGgsIHNjZW5lSGVpZ2h0KTtcblxuICAgIHRoaXMuaW5pdCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgICAgIGlmKHR5cGVvZihjYW52YXNDb250YWluZXIpID09PSAnc3RyaW5nJykge1xuICAgICAgICAgICAgY2FudmFzQ29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoY2FudmFzQ29udGFpbmVyKTtcbiAgICAgICAgfVxuICAgICAgICBpZih0eXBlb2YodGltZUNvbnRhaW5lcikgPT09ICdzdHJpbmcnKSB7XG4gICAgICAgICAgICB0aW1lQ29udGFpbmVyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQodGltZUNvbnRhaW5lcik7XG4gICAgICAgIH1cblxuICAgICAgICBjYW52YXNDb250YWluZXIuYXBwZW5kQ2hpbGQocmVuZGVyZXIudmlldyk7XG5cbiAgICAgICAgY29tcG9uZW50cy5mb3JFYWNoKGZ1bmN0aW9uKGMpe1xuICAgIFx0XHRjLmluaXQoKTtcbiAgICBcdH0pO1xuICAgIH07XG5cbiAgICB0aGlzLnJlZ2lzdGVyVGltZUNvbnRhaW5lciA9IGZ1bmN0aW9uKGNvbnRhaW5lcikge1xuICAgIFx0dGltZUNvbnRhaW5lci5wdXNoKGNvbnRhaW5lcik7XG4gICAgfTtcblxuICAgIHRoaXMucmVnaXN0ZXJDb21wb25lbnQgPSBmdW5jdGlvbihjb21wb25lbnQpIHtcbiAgICBcdGNvbXBvbmVudHMucHVzaChjb21wb25lbnQpO1xuICAgIFx0dGhpcy5zdGFnZS5hZGRDaGlsZChjb21wb25lbnQuY29udGFpbmVyKTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG4gICAgXHRjb21wb25lbnRzLmZvckVhY2goZnVuY3Rpb24oYyl7XG4gICAgXHRcdGMucmVmcmVzaCgpO1xuICAgIFx0fSk7XG4gICAgICAgIHJlbmRlcmVyLnJlbmRlcih0aGlzLnN0YWdlKTtcbiAgICB9O1xuXG4gICAgLy8gSW5pdCBwYWdlIGFuZCBpbnRlcnZhbHNcbiAgICB2YXIgcmVmcmVzaEludGVydmFsO1xuXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgICAgIGlmKCFleHRlcm5hbFJlZnJlc2gpIHtcbiAgICAgICAgICAgIHJlZnJlc2hJbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uKCkge190aGlzLnJlZnJlc2goKTt9LCAxMDAwL3RoaXMuZnJhbWVyYXRlKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbXBvbmVudHMuZm9yRWFjaChmdW5jdGlvbihjKXtcbiAgICBcdFx0Yy5zdGFydCgpO1xuICAgIFx0fSk7XG4gICAgfTtcblxuICAgIHRoaXMuc3RvcCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBpZighZXh0ZXJuYWxSZWZyZXNoKSB7XG4gICAgICAgICAgICBjbGVhckludGVydmFsKHJlZnJlc2hJbnRlcnZhbCk7XG4gICAgICAgIH1cbiAgICAgICAgXG4gICAgICAgIGNvbXBvbmVudHMuZm9yRWFjaChmdW5jdGlvbihjKXtcbiAgICBcdFx0Yy5zdG9wKCk7XG4gICAgXHR9KTtcbiAgICB9O1xuXG5cbiAgICB0aGlzLmxvZyA9IGZ1bmN0aW9uKG0pIHtcbiAgICAgICAgaWYodGhpcy5sb2dnZXIpIHtcbiAgICAgICAgICAgIHRoaXMubG9nZ2VyLmxvZyhtKTtcbiAgICAgICAgfVxuICAgIH07XG5cblxuICAgIHJldHVybiB0aGlzO1xufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBTdGFnZVZpZXc6IFN0YWdlVmlld1xufTtcbiIsIi8qKlxuKiBqcy91dGlscy5qc1xuKlxuKiBiYXNpYyB0b29sc1xuKlxuKi9cbi8qanNoaW50IGJpdHdpc2U6IGZhbHNlKi9cbi8qanNoaW50IGNhbWVsY2FzZTogZmFsc2UgKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG52YXIgbW9tZW50ID0gcmVxdWlyZSgnbW9tZW50Jyk7XG5cbmZ1bmN0aW9uIGZvcm1hdFRpbWUgKHRzKSB7XG4gICAgcmV0dXJuIG1vbWVudCh0cykuZm9ybWF0KCdISDptbTpzcycpO1xufVxuXG5mdW5jdGlvbiBjb2xvclRvSGV4KGMpIHtcbiAgICB2YXIgbSA9IC9yZ2JhP1xcKFxccyooXFxkKylcXHMqLFxccyooXFxkKylcXHMqLFxccyooXFxkKykvLmV4ZWMoYyk7XG4gICAgcmV0dXJuIG0gPyAnIycgKyAoMSA8PCAyNCB8IG1bMV0gPDwgMTYgfCBtWzJdIDw8IDggfCBtWzNdKS50b1N0cmluZygxNikuc3Vic3RyKDEpIDogYztcbn1cblxuZnVuY3Rpb24gZ2V0QW5ub3RDYXRlZ29yaWVzKHVybENhdGVnb3JpZXMsIGFubm90Q2F0ZWdvcmllcykge1xuXG4gICAgdmFyIGpzb25Mb2FkZXIgPSBuZXcgUElYSS5Kc29uTG9hZGVyKHVybENhdGVnb3JpZXMsIHRydWUpO1xuXG4gICAganNvbkxvYWRlci5vbignbG9hZGVkJywgZnVuY3Rpb24ocmVzKSB7XG4gICAgICAgIHZhciBkYXRhID0gcmVzLnRhcmdldC5qc29uO1xuXG4gICAgICAgIHdoaWxlKGFubm90Q2F0ZWdvcmllcy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICBhbm5vdENhdGVnb3JpZXMucG9wKCk7XG4gICAgICAgIH1cblxuICAgICAgICBkYXRhLnNlc3Npb25zLmZvckVhY2goZnVuY3Rpb24oc2Vzc2lvbikge1xuICAgICAgICAgICAgdmFyIGFubm90Q2F0ID0ge1xuICAgICAgICAgICAgICAgIHRzOiBzZXNzaW9uLnN0YXJ0X3RzID09PSBudWxsID8gbmV3IERhdGUoMCkgOiBEYXRlLnBhcnNlKHNlc3Npb24uc3RhcnRfdHMpLFxuICAgICAgICAgICAgICAgIGNvbG9yczoge31cbiAgICAgICAgICAgIH07XG4gICAgICAgICAgICB2YXIgY2F0ZWdvcmllc0pzb24gPSBzZXNzaW9uLmNhdGVnb3JpZXNfanNvbjtcbiAgICAgICAgICAgIGFubm90Q2F0Lm9yZGVyID0gY2F0ZWdvcmllc0pzb24ub3JkZXI7XG4gICAgICAgICAgICBpZiAodHlwZW9mKGFubm90Q2F0Lm9yZGVyWydkZWZhdWx0J10pID09PSAndW5kZWZpbmVkJyl7XG4gICAgICAgICAgICAgICAgYW5ub3RDYXQub3JkZXIucHVzaCgnZGVmYXVsdCcpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgdmFyIGNhdExpc3QgPSBfLmNsb25lKGNhdGVnb3JpZXNKc29uLmNhdGVnb3JpZXMpO1xuICAgICAgICAgICAgd2hpbGUoY2F0TGlzdC5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICAgICAgdmFyIGNhdCA9IGNhdExpc3QucG9wKCk7XG4gICAgICAgICAgICAgICAgaWYoY2F0LmNvZGUpIHtcbiAgICAgICAgICAgICAgICAgICAgYW5ub3RDYXQuY29sb3JzW2NhdC5jb2RlXSA9IGNvbG9yVG9IZXgoY2F0LmNvbG9yKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgaWYoY2F0LnN1YmNhdGVnb3JpZXMpIHtcbiAgICAgICAgICAgICAgICAgICAgY2F0TGlzdCA9IGNhdExpc3QuY29uY2F0KGNhdC5zdWJjYXRlZ29yaWVzKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjYXRlZ29yaWVzSnNvbi5jYXRlZ29yaWVzLmZvckVhY2goZnVuY3Rpb24oY2F0KSB7XG4gICAgICAgICAgICAgICAgaWYoY2F0LmNvZGUpIHtcbiAgICAgICAgICAgICAgICAgICAgYW5ub3RDYXQuY29sb3JzW2NhdC5jb2RlXSA9IGNvbG9yVG9IZXgoY2F0LmNvbG9yKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGFubm90Q2F0LmRlZmF1bHRDb2xvciA9IGNhdGVnb3JpZXNKc29uLmRlZmF1bHRDb2xvciB8fCAnIzUzNjk5MSc7XG4gICAgICAgICAgICBhbm5vdENhdGVnb3JpZXMucHVzaChhbm5vdENhdCk7XG4gICAgICAgIH0pO1xuICAgIH0pO1xuXG4gICAganNvbkxvYWRlci5sb2FkKCk7XG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgZm9ybWF0VGltZTogZm9ybWF0VGltZSxcbiAgICBnZXRBbm5vdENhdGVnb3JpZXM6IGdldEFubm90Q2F0ZWdvcmllcyxcbiAgICBjb2xvclRvSGV4OiBjb2xvclRvSGV4XG59O1xuIiwiLyoqXG4qIGpzL3dzd3JhcHBlci5qc1xuKlxuKiBzaW1wbGUgd2Vic2VydmljZSB3cmFwcGVyIHRvIHJlZ2lzdGVyIGNhbGxiYWNrcyBvbiBvbm1lc3NhZ2VcbipcbiovXG5cbi8qIGdsb2JhbCBXZWJTb2NrZXQ6IGZhbHNlICovXG5cbid1c2Ugc3RyaWN0JztcblxuZnVuY3Rpb24gV3NXcmFwcGVyKHdzdXJsLCBsb2dnZXIpIHtcblxuICAgIHZhciB1cmwgPSB3c3VybDtcbiAgICB2YXIgc29jayA9IG5ldyBXZWJTb2NrZXQodXJsKTtcbiAgICB2YXIgbG9nZ2VyT2JqID0gbG9nZ2VyO1xuXG4gICAgdmFyIGxvZyA9IGZ1bmN0aW9uKG1zZykge1xuICAgICAgICBpZihsb2dnZXJPYmopIHtcbiAgICAgICAgICAgIGxvZ2dlck9iai5sb2cobXNnKTtcbiAgICAgICAgfVxuICAgIH07XG5cbiAgICB2YXIgaGFuZGxlcnMgPSBbXTtcblxuICAgIHNvY2sub25vcGVuID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIGxvZygnQ29ubmVjdGVkIHRvICcgKyB1cmwpO1xuICAgIH07XG5cbiAgICBzb2NrLm9uY2xvc2UgPSBmdW5jdGlvbihlKSB7XG4gICAgICAgIGxvZygnQ29ubmVjdGlvbiBjbG9zZWQgKHdhc0NsZWFuID0gJyArIGUud2FzQ2xlYW4gKyAnLCBjb2RlID0gJyArIGUuY29kZSArICcsIHJlYXNvbiA9IFxcJycgKyBlLnJlYXNvbiArICdcXCcpJyk7XG4gICAgICAgIHNvY2sgPSBudWxsO1xuICAgIH07XG5cbiAgICBzb2NrLm9ubWVzc2FnZSA9IGZ1bmN0aW9uKGUpIHtcbiAgICAgICAgbG9nKCdyZWNlaXZlZCAnICsgZS5kYXRhKTtcbiAgICAgICAgdmFyIGRhdGEgPSBKU09OLnBhcnNlKGUuZGF0YSk7XG4gICAgICAgIGhhbmRsZXJzLmZvckVhY2goZnVuY3Rpb24oaGFuZGxlcikge1xuICAgICAgICAgICAgaGFuZGxlcihkYXRhKTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIHRoaXMubWVzc2FnZSA9IGZ1bmN0aW9uKGhhbmRsZXIpIHtcbiAgICAgICAgaWYoaGFuZGxlcikge1xuICAgICAgICAgICAgaGFuZGxlcnMucHVzaChoYW5kbGVyKTtcbiAgICAgICAgfVxuICAgIH07XG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgV3NXcmFwcGVyOiBXc1dyYXBwZXJcbn07XG4iXX0=