annot-server/static/js/annotviz.js
author rougeronj
Wed, 15 Apr 2015 18:26:59 +0200
changeset 149 3a74c554c9b9
parent 148 21634b2d7609
child 150 c8fe9606d967
permissions -rw-r--r--
remove top line

!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-01-23T09:30:00+01:00'),
        timeEnd: Date.parse('2015-01-23T18:30:00+01: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-01-22T09:30:00+01:00'),
        timeEnd: Date.parse('2015-01-22T18:30:00+01: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 defaultConfig = {
    orientation: 'horizontal',
    logger: undefined,
    sceneWidth: 1024,
    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;

    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(!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.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;

    this.start = function() {

        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,"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
*
*/

'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 channel = data.content[2];
        var sessionTs = data.content[1];

        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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5vZGVfbW9kdWxlcy9icm93c2VyaWZ5L25vZGVfbW9kdWxlcy9icm93c2VyLXBhY2svX3ByZWx1ZGUuanMiLCIuL2FwcC9qcy9tYWluLmpzIiwiL1VzZXJzL3JvdWdlcm9uai93b3Jrc3BhY2UvbW9ucy9jbGllbnQvYW5ub3R2aXovYXBwL2pzL2Fubm90c3JvbGwuanMiLCIvVXNlcnMvcm91Z2Vyb25qL3dvcmtzcGFjZS9tb25zL2NsaWVudC9hbm5vdHZpei9hcHAvanMvYW5ub3RzdGltZWxpbmUuanMiLCIvVXNlcnMvcm91Z2Vyb25qL3dvcmtzcGFjZS9tb25zL2NsaWVudC9hbm5vdHZpei9hcHAvanMvYW5ub3Rzdml6dmlldy5qcyIsIi9Vc2Vycy9yb3VnZXJvbmovd29ya3NwYWNlL21vbnMvY2xpZW50L2Fubm90dml6L2FwcC9qcy9kb3VibGVyb2xsLmpzIiwiL1VzZXJzL3JvdWdlcm9uai93b3Jrc3BhY2UvbW9ucy9jbGllbnQvYW5ub3R2aXovYXBwL2pzL2xvZ2dlci5qcyIsIi9Vc2Vycy9yb3VnZXJvbmovd29ya3NwYWNlL21vbnMvY2xpZW50L2Fubm90dml6L2FwcC9qcy9waWFub3JvbGwuanMiLCIvVXNlcnMvcm91Z2Vyb25qL3dvcmtzcGFjZS9tb25zL2NsaWVudC9hbm5vdHZpei9hcHAvanMvc3RhZ2V2aWV3LmpzIiwiL1VzZXJzL3JvdWdlcm9uai93b3Jrc3BhY2UvbW9ucy9jbGllbnQvYW5ub3R2aXovYXBwL2pzL3V0aWxzLmpzIiwiL1VzZXJzL3JvdWdlcm9uai93b3Jrc3BhY2UvbW9ucy9jbGllbnQvYW5ub3R2aXovYXBwL2pzL3dzd3JhcHBlci5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtBQ0FBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzlCQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQy9OQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzVRQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3pOQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUN0TUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDN0NBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUNoUkE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3BIQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDMUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsImZpbGUiOiJnZW5lcmF0ZWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uIGUodCxuLHIpe2Z1bmN0aW9uIHMobyx1KXtpZighbltvXSl7aWYoIXRbb10pe3ZhciBhPXR5cGVvZiByZXF1aXJlPT1cImZ1bmN0aW9uXCImJnJlcXVpcmU7aWYoIXUmJmEpcmV0dXJuIGEobywhMCk7aWYoaSlyZXR1cm4gaShvLCEwKTt2YXIgZj1uZXcgRXJyb3IoXCJDYW5ub3QgZmluZCBtb2R1bGUgJ1wiK28rXCInXCIpO3Rocm93IGYuY29kZT1cIk1PRFVMRV9OT1RfRk9VTkRcIixmfXZhciBsPW5bb109e2V4cG9ydHM6e319O3Rbb11bMF0uY2FsbChsLmV4cG9ydHMsZnVuY3Rpb24oZSl7dmFyIG49dFtvXVsxXVtlXTtyZXR1cm4gcyhuP246ZSl9LGwsbC5leHBvcnRzLGUsdCxuLHIpfXJldHVybiBuW29dLmV4cG9ydHN9dmFyIGk9dHlwZW9mIHJlcXVpcmU9PVwiZnVuY3Rpb25cIiYmcmVxdWlyZTtmb3IodmFyIG89MDtvPHIubGVuZ3RoO28rKylzKHJbb10pO3JldHVybiBzfSkiLCIvKipcbiAqIHNjcmlwdHMvbWFpbi5qc1xuICpcbiAqIFRoaXMgaXMgdGhlIHN0YXJ0aW5nIHBvaW50IGZvciB5b3VyIGFwcGxpY2F0aW9uLlxuICogVGFrZSBhIGxvb2sgYXQgaHR0cDovL2Jyb3dzZXJpZnkub3JnLyBmb3IgbW9yZSBpbmZvXG4gKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG52YXIgZG91Ymxlcm9sbCA9IHJlcXVpcmUoJy4vZG91Ymxlcm9sbCcpO1xudmFyIGFubm90c3JvbGwgPSByZXF1aXJlKCcuL2Fubm90c3JvbGwnKTtcbnZhciBhbm5vdHN0aW1lbGluZSA9IHJlcXVpcmUoJy4vYW5ub3RzdGltZWxpbmUnKTtcbnZhciBhbm5vdHN2aXp2aWV3ID0gcmVxdWlyZSgnLi9hbm5vdHN2aXp2aWV3Jyk7XG52YXIgc3RhZ2V2aWV3ID0gcmVxdWlyZSgnLi9zdGFnZXZpZXcnKTtcbnZhciB3c3dyYXBwZXIgPSByZXF1aXJlKCcuL3dzd3JhcHBlcicpO1xudmFyIGxvZ2dlciA9IHJlcXVpcmUoJy4vbG9nZ2VyJyk7XG52YXIgdXRpbHMgPSByZXF1aXJlKCcuL3V0aWxzJyk7XG5cbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG5cbm1vZHVsZS5leHBvcnRzID0gXyh7fSlcbiAgICAuZXh0ZW5kKGRvdWJsZXJvbGwpXG4gICAgLmV4dGVuZChhbm5vdHNyb2xsKVxuICAgIC5leHRlbmQoYW5ub3RzdGltZWxpbmUpXG4gICAgLmV4dGVuZChhbm5vdHN2aXp2aWV3KVxuICAgIC5leHRlbmQoc3RhZ2V2aWV3KVxuICAgIC5leHRlbmQod3N3cmFwcGVyKVxuICAgIC5leHRlbmQobG9nZ2VyKVxuICAgIC5leHRlbmQodXRpbHMpXG4gICAgLnZhbHVlKCk7XG4iLCIvKipcbioganMvYW5ub3RzUm9sbC5qc1xuKlxuKiBhbm5vdHNSb2xsIGJhc2ljIGNvbXBvbmVudFxuKlxuKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG5cbnZhciBERUZBVUxUX0FOTk9UX0NPTE9SID0gJyNiYWJhYmEnO1xuXG52YXIgZGVmYXVsdEFubm90U3R5bGVzID0ge1xuICAgICdsYWJlbCc6IHsgZm9udDogJzE2cHQgQXJpYWwgQm9sZCcsIGZpbGw6ICcjNjVBOTU0Jywgd29yZFdyYXA6IHRydWV9LFxuICAgICd0ZXh0JyA6IHsgZm9udDogJzEycHQgQXJpYWwgUmVndWxhcicsIGZpbGw6ICcjNDQ0NDQ0Jywgd29yZFdyYXA6IHRydWV9LFxuICAgICd1c2VyJyA6IHsgZm9udDogJzE0cHQgQXJpYWwgcmVndWxhcicsIGZpbGw6ICcjNjY2NjY2JyB9LFxufTtcblxudmFyIGRlZmF1bHRPcHRpb25zID0ge1xuICAgIGV4dGVybmFsUmVmcmVzaDogZmFsc2UsXG4gICAgZGVmYXVsdENvbG9yOiBERUZBVUxUX0FOTk9UX0NPTE9SLFxuICAgIGFubm90U3R5bGVzOiBkZWZhdWx0QW5ub3RTdHlsZXMsXG4gICAgaWdub3JlQW5ub3RzOmZhbHNlXG59O1xuXG5mdW5jdGlvbiBBbm5vdHNSb2xsKG9wdGlvbnMpIHtcblxuLy9wYXJlbnRDb250YWluZXIsIHhJbml0LCB5SW5pdCwgd2lkdGgsIGhlaWdodCwgd2lkdGhSb2xsLCBwaXhlbHNQZXJTZWNvbmQsIGFubm90Q29sb3JzXG4gICAgdmFyIF90aGlzID0gdGhpcztcbiAgICB2YXIgb3B0cyA9IF8ob3B0aW9ucykuZGVmYXVsdHMoZGVmYXVsdE9wdGlvbnMpLnZhbHVlKCk7XG5cblxuICAgIHRoaXMuY29udGFpbmVyID0gbmV3IFBJWEkuRGlzcGxheU9iamVjdENvbnRhaW5lcigpO1xuICAgIHRoaXMuY29udGFpbmVyLnggPSBvcHRzLnhJbml0O1xuICAgIHRoaXMuY29udGFpbmVyLnkgPSBvcHRzLnlJbml0O1xuICAgIHRoaXMuY29udGFpbmVyLndpZHRoID0gb3B0cy53aWR0aDtcblxuICAgIHRoaXMuaGVpZ2h0ID0gb3B0cy5oZWlnaHQ7XG4gICAgdGhpcy53aWR0aCA9IG9wdHMud2lkdGg7XG4gICAgdGhpcy53aWR0aFJvbGwgPSBvcHRzLndpZHRoUm9sbDtcbiAgICB0aGlzLnBpeGVsc1BlclNlY29uZCA9IG9wdHMucGl4ZWxzUGVyU2Vjb25kO1xuICAgIHRoaXMuYW5ub3RDb2xvcnMgPSBvcHRzLmFubm90Q29sb3JzO1xuICAgIHRoaXMuc3RhcnRUcyA9IG9wdHMuc3RhcnRUcyB8fCBEYXRlLm5vdygpO1xuICAgIHRoaXMuaWdub3JlQW5ub3RzID0gb3B0cy5pZ25vcmVBbm5vdHM7XG5cbiAgICB2YXIgeUluaXQgPSBvcHRzLnlJbml0O1xuICAgIHZhciBhbm5vdFN0eWxlcyA9IF8ob3B0cy5hbm5vdFN0eWxlcykuZGVmYXVsdHMoZGVmYXVsdEFubm90U3R5bGVzKS52YWx1ZSgpO1xuICAgIHZhciBtYXJnaW5YID0gMTU7XG4gICAgZm9yKHZhciBzdHlsZSBpbiBhbm5vdFN0eWxlcykge1xuICAgIFx0aWYgKGFubm90U3R5bGVzW3N0eWxlXS53b3JkV3JhcCA9PT0gdHJ1ZSl7XG4gICAgXHRcdGFubm90U3R5bGVzW3N0eWxlXS53b3JkV3JhcFdpZHRoID0gdGhpcy53aWR0aFJvbGwgLSBtYXJnaW5YO1xuICAgIFx0fVxuICAgIH1cbiAgICB2YXIgc3RhcnRlZCA9IGZhbHNlO1xuICAgIHZhciB3cyA9IG9wdHMud3M7XG4gICAgdmFyIGV4dGVybmFsUmVmcmVzaCA9IG9wdHMuZXh0ZXJuYWxSZWZyZXNoO1xuICAgIHZhciBzdGFnZVZpZXcgPSBvcHRzLnN0YWdlVmlldztcbiAgICB2YXIgd2FpdEludGVydmFsO1xuICAgIHZhciB3YWl0ID0gMDtcblxuICAgIHN0YWdlVmlldy5yZWdpc3RlckNvbXBvbmVudCh0aGlzKTtcblxuICAgIHZhciBpc0hpZGRlbiA9IGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgICAgIC8vIFRPRE86IHRoZSBvcmlnaW4gcG9pbnQgaXMgYW4gYXBwcm94aW1hdGlvbi4gU2hvdWxkIHJlZmluZSB0aGlzXG4gICAgICAgIHZhciBnbG9iYWxQb3MgPSBjaGlsZC50b0dsb2JhbChuZXcgUElYSS5Qb2ludCgwLDApKTtcbiAgICAgICAgcmV0dXJuICgoZ2xvYmFsUG9zLnggKyBjaGlsZC53aWR0aCkgPCAwKSB8fCAoKGdsb2JhbFBvcy55ICsgY2hpbGQuaGVpZ2h0KSA8IDApIDtcbiAgICB9O1xuXG4gICAgdGhpcy5hZGRBbm5vdHMgPSBmdW5jdGlvbihkYXRhKSB7XG5cbiAgICAgICAgLy92YXIgdGl0bGUgPSBkYXRhLmNvbnRlbnQuY2F0ZWdvcnkubGFiZWw7XG4gICAgICAgIC8vdmFyIHVzZXIgPSBkYXRhLmNvbnRlbnQudXNlcjtcbiAgICAgICAgLy9UZXN0IGNhdCBhbmQgY29sb3JcbiAgICAgICAgLy92YXIgY29sb3JBbm5vdCA9IDB4NjVBOTU0O1xuICAgICAgICB2YXIgY2F0ZWdvcnkgPSBkYXRhLmNvbnRlbnQuY2F0ZWdvcnkubGFiZWwsXG4gICAgICAgICAgICB0ZXh0ICAgICA9IGRhdGEuY29udGVudC50ZXh0LFxuICAgICAgICAgICAgdXNlciAgICAgPSBkYXRhLmNvbnRlbnQudXNlcixcbiAgICAgICAgICAgIHRzICAgICAgID0gRGF0ZS5wYXJzZShkYXRhLnRzKSxcbiAgICAgICAgICAgIGNvbG9yICAgID0gZGF0YS5jb250ZW50LmNvbG9yIHx8IHRoaXMuZ2V0Q29sb3IodHMsIGRhdGEuY29udGVudC5jYXRlZ29yeS5jb2RlKTtcblxuICAgICAgICB0aGlzLmFkZEFubm90KGNhdGVnb3J5LCB0ZXh0LCB1c2VyLCBjb2xvciwgdHMpO1xuICAgIH07XG5cbiAgICB0aGlzLmdldENvbG9yID0gZnVuY3Rpb24odHMsIGNvZGUpIHtcbiAgICAgICAgdmFyIGNvbG9yc0RlZjtcbiAgICAgICAgXyh0aGlzLmFubm90Q29sb3JzKS5lYWNoUmlnaHQoZnVuY3Rpb24oY2RlZikge1xuICAgICAgICAgICAgaWYoY2RlZi50cyA8IHRzKSB7XG4gICAgICAgICAgICAgICAgY29sb3JzRGVmID0gY2RlZi5jb2xvcnM7XG4gICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICAgICAgfVxuICAgICAgICB9KTtcbiAgICAgICAgdmFyIHJlc0NvbG9yO1xuICAgICAgICBpZihjb2xvcnNEZWYpIHtcbiAgICAgICAgICAgIHJlc0NvbG9yID0gY29sb3JzRGVmW2NvZGVdO1xuICAgICAgICB9XG4gICAgICAgIGlmKCFyZXNDb2xvcikge1xuICAgICAgICAgICAgcmVzQ29sb3IgPSBjb2xvcnNEZWYuZGVmYXVsdENvbG9yIHx8IERFRkFVTFRfQU5OT1RfQ09MT1I7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIHJlc0NvbG9yO1xuICAgIH07XG5cbiAgICB0aGlzLmFkZEFubm90ID0gZnVuY3Rpb24oY2F0ZWdvcnksIHRleHQsIHVzZXIsIGNvbG9yLCB0cyl7XG5cbiAgICAgICAgdmFyIHggPSAwO1xuICAgICAgICB2YXIgeSA9ICh0cy10aGlzLnN0YXJ0VHMpICogdGhpcy5waXhlbHNQZXJTZWNvbmQgLyAxMDAwICsgeUluaXQ7XG5cbiAgICAgICAgdmFyIGNvbG9ySGV4ID0gcGFyc2VJbnQoY29sb3IucmVwbGFjZSgvXiMvLCAnJyksIDE2KTtcblxuICAgICAgICBpZiAod2FpdCA9PT0gMCl7XG5cdCAgICAgICAgdmFyIGdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKVxuXHQgICAgICAgICAgICAuYmVnaW5GaWxsKGNvbG9ySGV4KVxuXHQgICAgICAgICAgICAuZHJhd1JlY3QoeCwgeSwgMTAsIDMpXG5cdCAgICAgICAgICAgIC5lbmRGaWxsKCk7XG5cblx0ICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChncmFwaGljcyk7XG5cblx0ICAgICAgICB2YXIgdGV4dEhlaWdodCA9IDA7XG5cdCAgICAgICAgdmFyIGNhdExhYmVsID0gbmV3IFBJWEkuVGV4dChcblx0ICAgICAgICAgICAgY2F0ZWdvcnksXG5cdCAgICAgICAgICAgIF8oYW5ub3RTdHlsZXMubGFiZWwpLmV4dGVuZCh7ZmlsbDogY29sb3J9KS52YWx1ZSgpXG5cdCAgICAgICAgKTtcblx0ICAgICAgICBjYXRMYWJlbC54ID0geCArIG1hcmdpblg7XG5cdCAgICAgICAgY2F0TGFiZWwueSA9IHkgLSAyMztcblx0ICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChjYXRMYWJlbCk7XG5cdCAgICAgICAgdGV4dEhlaWdodCArPSAoY2F0TGFiZWwuaGVpZ2h0IC0gMjMgKyAyKTtcblxuXHQgICAgICAgIGlmKHRleHQpIHtcblx0ICAgICAgICAgICAgdmFyIGNhdFRleHQgPSBuZXcgUElYSS5UZXh0KHRleHQsIGFubm90U3R5bGVzLnRleHQpO1xuXHQgICAgICAgICAgICBjYXRUZXh0LnggPSB4ICsgbWFyZ2luWDtcblx0ICAgICAgICAgICAgY2F0VGV4dC55ID0geSArIHRleHRIZWlnaHQ7XG5cdCAgICAgICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGNhdFRleHQpO1xuXHQgICAgICAgICAgICB0ZXh0SGVpZ2h0ICs9IChjYXRUZXh0LmhlaWdodCArIDIpO1xuXHQgICAgICAgIH1cblxuXHQgICAgICAgIHZhciBjYXRVc2VyID0gbmV3IFBJWEkuVGV4dCh1c2VyLCBhbm5vdFN0eWxlcy51c2VyKTtcblx0ICAgICAgICBjYXRVc2VyLnggPSB4ICsgbWFyZ2luWDtcblx0ICAgICAgICBjYXRVc2VyLnkgPSB5ICsgdGV4dEhlaWdodDtcblx0ICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChjYXRVc2VyKTtcblx0ICAgICAgICB0ZXh0SGVpZ2h0ICs9IChjYXRVc2VyLmhlaWdodCArIDgpO1xuXG5cdCAgICAgICAgaWYgKHRoaXMuaWdub3JlQW5ub3RzID09PSB0cnVlKXtcblx0XHQgICAgICAgIHdhaXQgPSB0ZXh0SGVpZ2h0IC8gdGhpcy5waXhlbHNQZXJTZWNvbmQ7XG5cdFx0ICAgICAgICB3YWl0SW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbigpIHtfdGhpcy5yZWZyZXNoV2FpdCgpO30sIDEwMDApO1xuXHQgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuYWRkQW5ub3RMaW5lKGNvbG9ySGV4LCB5KTtcbiAgICB9O1xuXG4gICAgdGhpcy5hZGRBbm5vdExpbmUgPSBmdW5jdGlvbihjb2xvciwgeSkge1xuICAgICAgICB2YXIgeCA9IHRoaXMud2lkdGhSb2xsO1xuXG5cbiAgICAgICAgdmFyIGdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKVxuICAgICAgICAgICAgLmJlZ2luRmlsbChjb2xvcilcbiAgICAgICAgICAgIC5kcmF3UmVjdCh4LCB5LCB0aGlzLndpZHRoIC0geCwgMylcbiAgICAgICAgICAgIC5lbmRGaWxsKCk7XG5cbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoZ3JhcGhpY3MpO1xuICAgIH07XG5cbiAgICB0aGlzLm1vdmVUbyA9IGZ1bmN0aW9uKGRpZmZUaW1lKXtcbiAgICBcdHRoaXMuY29udGFpbmVyLnkgPSBNYXRoLmZsb29yKGRpZmZUaW1lKnRoaXMucGl4ZWxzUGVyU2Vjb25kKTtcbiAgICB9O1xuXG4gICAgdGhpcy5tb3ZlID0gdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIHZhciBkaWZmID0gKHRoaXMuc3RhcnRUcyAtIERhdGUubm93KCkpLzEwMDA7XG4gICAgICAgIHRoaXMubW92ZVRvKGRpZmYpO1xuICAgIH07XG5cbiAgICB0aGlzLnJlZnJlc2hXYWl0ID0gZnVuY3Rpb24oKXtcbiAgICBcdHdhaXQgLT0gMTtcbiAgICBcdGlmICh3YWl0IDwgMCl7XG4gICAgXHRcdHdhaXQgPSAwO1xuICAgIFx0XHRjbGVhckludGVydmFsKHdhaXRJbnRlcnZhbCk7XG4gICAgXHR9XG4gICAgfTtcblxuICAgIHRoaXMucmVtb3ZlUGFzc2VkT2JqZXRzID0gZnVuY3Rpb24oKXtcbiAgICAgICAgdmFyIGNoaWxkcmVuVG9SZW1vdmUgPSBbXTtcbiAgICAgICAgXyhfdGhpcy5jb250YWluZXIuY2hpbGRyZW4pLmZvckVhY2goZnVuY3Rpb24oY2hpbGQpIHtcbiAgICAgICAgICAgIHJldHVybiB0eXBlb2YoY2hpbGQpID09PSAndW5kZWZpbmVkJyB8fFxuICAgICAgICAgICAgICAgIChpc0hpZGRlbihjaGlsZCkgJiYgY2hpbGRyZW5Ub1JlbW92ZS5wdXNoKGNoaWxkKSk7XG4gICAgICAgIH0pO1xuICAgICAgICBjaGlsZHJlblRvUmVtb3ZlLmZvckVhY2goZnVuY3Rpb24oY2hpbGQpIHtcbiAgICAgICAgICAgIF90aGlzLmNvbnRhaW5lci5yZW1vdmVDaGlsZChjaGlsZCk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cbiAgICB0aGlzLmluaXQgPSBmdW5jdGlvbigpIHtcblxuICAgICAgICB3cy5tZXNzYWdlKGZ1bmN0aW9uKGRhdGEpIHtcbiAgICAgICAgICAgIF90aGlzLmFkZEFubm90cyhkYXRhKTtcbiAgICAgICAgfSk7XG5cbiAgICB9O1xuXG5cbiAgICB0aGlzLnN0YXJ0ID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIGlmKCFzdGFydGVkKSB7XG4gICAgICAgICAgICB0aGlzLnN0YXJ0VHMgPSBEYXRlLm5vdygpO1xuICAgICAgICAgICAgc3RhcnRlZCA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5jbGVhbkludGVydmFsID0gc2V0SW50ZXJ2YWwoZnVuY3Rpb24gKCkgeyBfdGhpcy5yZW1vdmVQYXNzZWRPYmpldHMoKTsgfSwgMTAwMCAqIHRoaXMuaGVpZ2h0IC8gdGhpcy5waXhlbHNQZXJTZWNvbmQgKTtcbiAgICAgICAgaWYoIWV4dGVybmFsUmVmcmVzaCkge1xuICAgICAgICAgICAgdGhpcy5yZWZyZXNoSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbigpIHtfdGhpcy5tb3ZlKCk7fSwgMTAwMC90aGlzLmZyYW1lcmF0ZSk7XG4gICAgICAgIH1cbiAgICB9O1xuXG4gICAgdGhpcy5zdG9wID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIGNsZWFySW50ZXJ2YWwodGhpcy5jbGVhbkludGVydmFsKTtcbiAgICAgICAgaWYoIWV4dGVybmFsUmVmcmVzaCkge1xuICAgICAgICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLnJlZnJlc2hJbnRlcnZhbCk7XG4gICAgICAgIH1cbiAgICB9O1xuXG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIEFubm90c1JvbGw6IEFubm90c1JvbGwsXG59O1xuIiwiLyoqXG4qIGpzL2Fubm90c3RpbWVsaW5lXG4qXG4qIGFubm90c3RpbWVsaW5lIGJhc2ljIGNvbXBvbmVudFxuKlxuKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciBVdGlscyA9IHJlcXVpcmUoJy4vdXRpbHMuanMnKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG5cbnZhciBkZWZhdWx0T3B0aW9ucyA9IHtcbiAgICBsb2dnZXI6IHVuZGVmaW5lZCxcbiAgICBpbnRlcnZhbFdpZHRoOiAxMCxcbiAgICBpbnRlcnZhbEhlaWdodDogNSxcbiAgICBtYXhDZWxsSGVpZ2h0OiAyMDAsXG4gICAgcmFkaXVzOiAzMDAsXG4gICAgc2VydmVyVXJsOiAnaHR0cDovLzEyNy4wLjAuMTo4MDgwJyxcbiAgICBjaGFubmVsOiAnQU5OT1QnLFxuICAgIG1heFBhZ2VzOiAxMDAwLFxuICAgIHNob3dDbG9ja0dyYXBoaWNzOiB0cnVlLFxuICAgIGFyY2hpdmU6IGZhbHNlXG59O1xuXG5cbmZ1bmN0aW9uIEFubm90c1RpbWVMaW5lKG9wdGlvbnMpe1xuICAgIHZhciBfdGhpcyA9IHRoaXM7XG4gICAgdmFyIG9wdHMgPSBfKG9wdGlvbnMpLmRlZmF1bHRzKGRlZmF1bHRPcHRpb25zKS52YWx1ZSgpO1xuXG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IG9wdHMueEluaXQ7XG4gICAgdGhpcy5jb250YWluZXIueSA9IG9wdHMueUluaXQ7XG4gICAgdGhpcy5jb250YWluZXIud2lkdGggPSBvcHRzLndpZHRoO1xuICAgIHRoaXMuY29udGFpbmVyLmhlaWdodCA9IG9wdHMuaGVpZ2h0O1xuXG4gICAgdGhpcy50aW1lQmVnaW4gPSBvcHRzLnRpbWVCZWdpbjtcbiAgICB0aGlzLnRpbWVFbmQgPSBvcHRzLnRpbWVFbmQ7XG4gICAgdGhpcy5kdXJhdGlvbiA9ICh0aGlzLnRpbWVFbmQgLSB0aGlzLnRpbWVCZWdpbikvMTAwMDtcbiAgICB0aGlzLndpZHRoID0gb3B0cy53aWR0aDtcbiAgICB0aGlzLmhlaWdodCA9IG9wdHMuaGVpZ2h0O1xuICAgIHRoaXMuaW50ZXJ2YWxIZWlnaHQgPSBvcHRzLmludGVydmFsSGVpZ2h0O1xuICAgIHRoaXMuaW50ZXJ2YWxXaWR0aCA9IG9wdHMuaW50ZXJ2YWxXaWR0aDtcbiAgICB0aGlzLm1heENlbGxIZWlnaHQgPSBvcHRzLm1heENlbGxIZWlnaHQ7XG4gICAgdGhpcy5hbm5vdENhdGVnb3JpZXMgPSBvcHRzLmFubm90Q2F0ZWdvcmllcztcbiAgICB0aGlzLnN0YXJ0VHMgPSBvcHRpb25zLnN0YXJ0VHMgfHwgRGF0ZS5ub3coKTtcbiAgICB0aGlzLnNob3dDbG9ja0dyYXBoaWNzID0gb3B0cy5zaG93Q2xvY2tHcmFwaGljcztcbiAgICB0aGlzLmFyY2hpdmUgPSBvcHRzLmFyY2hpdmU7XG5cbiAgICB0aGlzLmNpcmNsZVggPSBvcHRzLmNpcmNsZVggfHwgKHRoaXMud2lkdGgvMik7XG4gICAgdGhpcy5jaXJjbGVZID0gb3B0cy5jaXJjbGVZIHx8ICh0aGlzLmhlaWdodC8yKTtcbiAgICB0aGlzLnJhZGl1cyA9IG9wdHMucmFkaXVzO1xuICAgIHZhciBwZXJpbWV0ZXIgPSAyKk1hdGguUEkqIHRoaXMucmFkaXVzO1xuICAgIHRoaXMuaW50ZXJ2YWxEdXJhdGlvbiA9ICh0aGlzLmludGVydmFsV2lkdGggKiB0aGlzLmR1cmF0aW9uIC8gcGVyaW1ldGVyKTtcblxuICAgIHZhciBjaGFubmVsID0gb3B0cy5jaGFubmVsO1xuICAgIHZhciBldmVudENvZGUgPSBvcHRzLmV2ZW50Q29kZTtcbiAgICB2YXIgc2VydmVyVXJsID0gb3B0cy5zZXJ2ZXJVcmw7XG4gICAgdmFyIG1heFBhZ2VzID0gb3B0cy5tYXhQYWdlcztcblxuICAgIHZhciB0b3RhbEluZGV4ID0gTWF0aC5mbG9vciggcGVyaW1ldGVyL3RoaXMuaW50ZXJ2YWxXaWR0aCk7XG5cbiAgICB0aGlzLmNlbGxzID0gW107XG4gICAgZm9yICh2YXIgaT0wOyBpPChwZXJpbWV0ZXIvdGhpcy5pbnRlcnZhbFdpZHRoKSA7IGkrKyl7XG4gICAgICAgIHRoaXMuY2VsbHNbaV0gPSBbXTtcbiAgICAgICAgdGhpcy5jZWxsc1tpXS5pID0gaTtcbiAgICAgICAgdGhpcy5jZWxsc1tpXS50b3RhbEFubm90cyA9IDA7XG4gICAgICAgIHRoaXMuY2VsbHNbaV0uY2F0ZWdvcmllcyA9IHt9O1xuICAgIH1cblxuICAgIHZhciB3cyA9IG9wdHMud3M7XG4gICAgdmFyIHN0YWdlVmlldyA9IG9wdHMuc3RhZ2VWaWV3O1xuXG4gICAgLy9kcmF3IHRoZSBiYXNlIC0gY2lyY2xlIGFuZCBsaW5lIHRvIGxvY2F0ZSB0aGUgc2NlbmVcbiAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpO1xuICAgIGdyYXBoaWNzLmxpbmVTdHlsZSgyLCAweDY0NjQ2NClcbiAgICAgICAgLmRyYXdDaXJjbGUodGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVksIHRoaXMucmFkaXVzIC0gMylcbiAgICAgICAgLmVuZEZpbGwoKTtcbiAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChncmFwaGljcyk7XG5cbiAgICBzdGFnZVZpZXcucmVnaXN0ZXJDb21wb25lbnQodGhpcyk7XG5cbiAgICB2YXIgbG9hZEFyY2hpdmVzID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIC8vc3RhcnQgdGltZUJlZ2luIGVuZCBzdGFydFRpbWVcbiAgICAgICAgLy9xdWVyeSAtPiBuZWVkIGNoYW5uZWwgKyBldmVudENvZGVcbiAgICAgICAgLy9pdGVyYXRlIG92ZXIgZGF0YSBmaWxsIGNlbGxzXG4gICAgICAgIHZhciBzdGFydFRzID0gX3RoaXMudGltZUJlZ2luO1xuICAgICAgICB2YXIgZW5kVHMgPSBNYXRoLm1pbihfdGhpcy50aW1lRW5kLF90aGlzLnN0YXJ0VHMpO1xuXG4gICAgICAgIHZhciB1cmwgPSBzZXJ2ZXJVcmwgKyAnL3AvYXBpL3YxL2Fubm90YXRpb24nO1xuICAgICAgICB2YXIgZmlsdGVycyA9IFtcbiAgICAgICAgICAgIHsgbmFtZTogJ3RzJywgb3A6ICc+JywgdmFsOiBuZXcgRGF0ZShzdGFydFRzKS50b0lTT1N0cmluZygpfSwgLy9zdGFydFxuICAgICAgICAgICAgeyBuYW1lOiAndHMnLCBvcDogJzw9JywgdmFsOiBuZXcgRGF0ZShlbmRUcykudG9JU09TdHJpbmcoKX0sIC8vZW5kXG4gICAgICAgICAgICB7IG5hbWU6ICdjaGFubmVsJywgb3A6ICc9PScsIHZhbDogY2hhbm5lbH0sIC8vY2hhbm5lbFxuICAgICAgICAgICAgeyBuYW1lOiAnZXZlbnRfY29kZScsIG9wOiAnPT0nLCB2YWw6IGV2ZW50Q29kZX0gLy9ldmVudGNvZGVcbiAgICAgICAgXTtcblxuICAgICAgICB1cmwgPSB1cmwgKyAnP3E9JyArIEpTT04uc3RyaW5naWZ5KHtmaWx0ZXJzOmZpbHRlcnN9KTtcblxuICAgICAgICB2YXIgdG90YWxQYWdlID0gMTtcbiAgICAgICAgdmFyIGN1cnJlbnRQYWdlID0gMTtcblxuICAgICAgICB2YXIgcHJvY2Vzc1Jlc0Z1bmN0aW9uID0gZnVuY3Rpb24ocmVzKSB7XG4gICAgICAgICAgICAvL2NvbnNvbGUubG9nKFwiUkVTIGFyY2hpdmVcIiwgcmVzKTtcbiAgICAgICAgICAgIGlmKHJlcykge1xuICAgICAgICAgICAgICAgIHZhciBkYXRhID0gcmVzLnRhcmdldC5qc29uO1xuICAgICAgICAgICAgICAgIC8qanNoaW50IC1XMDY5ICovXG4gICAgICAgICAgICAgICAgdG90YWxQYWdlID0gTWF0aC5taW4obWF4UGFnZXMscGFyc2VJbnQoZGF0YVsndG90YWxfcGFnZXMnXSkpO1xuICAgICAgICAgICAgICAgIGRhdGEub2JqZWN0cy5mb3JFYWNoKGZ1bmN0aW9uKGFubm90YXRpb24pIHtcbiAgICAgICAgICAgICAgICAgICAgX3RoaXMuYWRkQW5ub3QoYW5ub3RhdGlvbik7XG4gICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBpZihjdXJyZW50UGFnZSA8PSB0b3RhbFBhZ2UpIHtcbiAgICAgICAgICAgICAgICB2YXIganNvbkxvYWRlciA9IG5ldyBQSVhJLkpzb25Mb2FkZXIodXJsKycmcGFnZT0nK2N1cnJlbnRQYWdlLCB0cnVlKTtcbiAgICAgICAgICAgICAgICBqc29uTG9hZGVyLm9uKCdsb2FkZWQnLCBwcm9jZXNzUmVzRnVuY3Rpb24pO1xuICAgICAgICAgICAgICAgIGpzb25Mb2FkZXIubG9hZCgpO1xuICAgICAgICAgICAgICAgIGN1cnJlbnRQYWdlKys7XG4gICAgICAgICAgICB9XG4gICAgICAgIH07XG4gICAgICAgIHByb2Nlc3NSZXNGdW5jdGlvbigpO1xuXG4gICAgfTtcblxuICAgIC8vQWRkIEFubm90YXRpb24gdG8gdGhlIFRpbWVMaW5lXG4gICAgdGhpcy5hZGRBbm5vdCA9IGZ1bmN0aW9uKGRhdGEpe1xuXG4gICAgICAgIHZhciB0cyA9IERhdGUucGFyc2UoZGF0YS50cyk7XG4gICAgICAgIHZhciBjb2xvcnNEZWY7XG4gICAgICAgIF8odGhpcy5hbm5vdENhdGVnb3JpZXMpLmVhY2hSaWdodChmdW5jdGlvbihjZGVmKSB7XG4gICAgICAgICAgICBpZihjZGVmLnRzIDwgdHMpIHtcbiAgICAgICAgICAgICAgICBjb2xvcnNEZWYgPSBjZGVmO1xuICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cbiAgICAgICAgaWYoIWNvbG9yc0RlZikge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHRoaXMudGltZUVuZCA+IHRzKXtcbiAgICAgICAgICAgIHZhciBpID0gTWF0aC5mbG9vcigodHMgLSB0aGlzLnRpbWVCZWdpbikvKDEwMDAqdGhpcy5pbnRlcnZhbER1cmF0aW9uKSk7XG5cbiAgICAgICAgICAgIGlmICh0eXBlb2YodGhpcy5jZWxsc1tpXS5ncmFwaGljcykgPT09ICd1bmRlZmluZWQnKXtcbiAgICAgICAgICAgICAgICB0aGlzLmluaXRDZWxsKHRoaXMuY2VsbHNbaV0sIGNvbG9yc0RlZik7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIHZhciBhbm5vdENvZGU7XG4gICAgICAgICAgICBpZiAodHlwZW9mKGNvbG9yc0RlZi5jb2xvcnNbZGF0YS5jb250ZW50LmNhdGVnb3J5LmNvZGVdKSAhPT0gJ3VuZGVmaW5lZCcpe1xuICAgICAgICAgICAgICAgIGFubm90Q29kZSA9IGRhdGEuY29udGVudC5jYXRlZ29yeS5jb2RlO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICBhbm5vdENvZGUgPSAnZGVmYXVsdCc7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIHRoaXMuY2VsbHNbaV0uY2F0ZWdvcmllc1thbm5vdENvZGVdLmNvdW50ICs9IDE7XG4gICAgICAgICAgICB0aGlzLmNlbGxzW2ldLnRvdGFsQW5ub3RzICs9MTtcbiAgICAgICAgICAgIHRoaXMucmVkcmF3Q2VsbCh0aGlzLmNlbGxzW2ldLCBjb2xvcnNEZWYpO1xuICAgICAgICB9XG4gICAgfTtcblxuICAgIHRoaXMuaW5pdENsb2NrR3JhcGhpY3MgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgdmFyIHRCZWcgPSBuZXcgUElYSS5UZXh0KFV0aWxzLmZvcm1hdFRpbWUodGhpcy50aW1lQmVnaW4pLCB7IGZvbnQ6ICcxMnB0IEdvdGhpYyBTdGFuZGFyZCcsIGZpbGw6ICcjNjQ2NDY0JyB9KTtcbiAgICAgICAgdEJlZy54ID0gdGhpcy5jaXJjbGVYICsgMTU7XG4gICAgICAgIHRCZWcueSA9IHRoaXMuY2lyY2xlWSAtIHRoaXMucmFkaXVzIC0gdGhpcy5tYXhDZWxsSGVpZ2h0IC0gMTA7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKHRCZWcpO1xuXG4gICAgICAgIHZhciB0RW5kID0gbmV3IFBJWEkuVGV4dChVdGlscy5mb3JtYXRUaW1lKHRoaXMudGltZUVuZCksIHsgZm9udDogJzEycHQgR290aGljIFN0YW5kYXJkJywgZmlsbDogJyM2NDY0NjQnIH0pO1xuICAgICAgICB0RW5kLnggPSB0aGlzLmNpcmNsZVggLSAxNSAtIHRFbmQud2lkdGg7XG4gICAgICAgIHRFbmQueSA9IHRoaXMuY2lyY2xlWSAtIHRoaXMucmFkaXVzIC0gdGhpcy5tYXhDZWxsSGVpZ2h0IC0gMTA7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKHRFbmQpO1xuXG4gICAgICAgIHZhciB0MTUgPSBuZXcgUElYSS5UZXh0KFV0aWxzLmZvcm1hdFRpbWUoKCh0aGlzLnRpbWVFbmQgLSB0aGlzLnRpbWVCZWdpbikvNCkgKyB0aGlzLnRpbWVCZWdpbiksIHsgZm9udDogJzEycHQgR290aGljIFN0YW5kYXJkJywgZmlsbDogJyM2NDY0NjQnIH0pO1xuICAgICAgICB0MTUueCA9IHRoaXMuY2lyY2xlWCArIHRoaXMucmFkaXVzICsgdGhpcy5tYXhDZWxsSGVpZ2h0ICsgMTAgO1xuICAgICAgICB0MTUueSA9IHRoaXMuY2lyY2xlWSAtIHQxNS5oZWlnaHQ7XG4gICAgICAgIHQxNS5yb3RhdGlvbiA9IE1hdGguUEkgLzI7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKHQxNSk7XG5cbiAgICAgICAgdmFyIHQzMCA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSgoKHRoaXMudGltZUVuZCAtIHRoaXMudGltZUJlZ2luKS8yKSArIHRoaXMudGltZUJlZ2luKSwgeyBmb250OiAnMTJwdCBHb3RoaWMgU3RhbmRhcmQnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG4gICAgICAgIHQzMC54ID0gdGhpcy5jaXJjbGVYIC0gdDMwLndpZHRoLzI7XG4gICAgICAgIHQzMC55ID0gdGhpcy5jaXJjbGVZICsgdGhpcy5yYWRpdXMgKyB0aGlzLm1heENlbGxIZWlnaHQgLSAyO1xuICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZCh0MzApO1xuXG4gICAgICAgIHZhciB0NDUgPSBuZXcgUElYSS5UZXh0KFV0aWxzLmZvcm1hdFRpbWUoKCh0aGlzLnRpbWVFbmQgLSB0aGlzLnRpbWVCZWdpbikqMy80KSArIHRoaXMudGltZUJlZ2luKSwgeyBmb250OiAnMTJwdCBHb3RoaWMgU3RhbmRhcmQnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG4gICAgICAgIHQ0NS54ID0gdGhpcy5jaXJjbGVYIC0gdGhpcy5yYWRpdXMgLSB0aGlzLm1heENlbGxIZWlnaHQgLSAxMCA7XG4gICAgICAgIHQ0NS55ID0gdGhpcy5jaXJjbGVZICsgdDE1LmhlaWdodDtcbiAgICAgICAgdDQ1LnJvdGF0aW9uID0gLU1hdGguUEkvMjtcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodDQ1KTtcblxuICAgICAgICB2YXIgbGluZVYgPSBuZXcgUElYSS5HcmFwaGljcygpO1xuICAgICAgICBsaW5lVi5saW5lU3R5bGUoMSwgMHg2NDY0NjQpXG4gICAgICAgICAgICAubW92ZVRvKHRoaXMuY2lyY2xlWCwgdGhpcy5jaXJjbGVZIC0gKHRoaXMucmFkaXVzLzMpLzIpXG4gICAgICAgICAgICAubGluZVRvKHRoaXMuY2lyY2xlWCwgdGhpcy5jaXJjbGVZIC0gdGhpcy5yYWRpdXMgLSB0aGlzLm1heENlbGxIZWlnaHQgLSAxMClcbiAgICAgICAgICAgIC5lbmRGaWxsKCk7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGxpbmVWKTtcbiAgICB9O1xuXG4gICAgLy9EcmF3IHRoZSBjZWxsdWxlXG4gICAgdGhpcy5yZWRyYXdDZWxsID0gZnVuY3Rpb24oY2VsbCwgY29sb3JzRGVmKXtcbiAgICAgICAgdmFyIHkgPSAwO1xuXG4gICAgICAgIC8vQ2hlY2sgaWYgdG90YWwgaGVpZ2h0IGlzIGhpZ2hlciB0aGFuIE1heCBDZWxsIEhlaWdodFxuICAgICAgICB2YXIgaGVpZ2h0U3RlcDtcbiAgICAgICAgaWYgKChjZWxsLnRvdGFsQW5ub3RzKnRoaXMuaW50ZXJ2YWxIZWlnaHQpID4gdGhpcy5tYXhDZWxsSGVpZ2h0KXtcbiAgICAgICAgICAgIGhlaWdodFN0ZXAgPSB0aGlzLm1heENlbGxIZWlnaHQvY2VsbC50b3RhbEFubm90cztcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGhlaWdodFN0ZXAgPSB0aGlzLmludGVydmFsSGVpZ2h0O1xuICAgICAgICB9XG5cbiAgICAgICAgLy9EcmF3IHRoZSByZWN0IGRlcGVuZGluZyBvbiB0aGUgaGVpZ2h0IHN0ZXAgY2FsY3VsYXRlZFxuICAgICAgICBmb3IgKHZhciBpPTA7IGk8IGNvbG9yc0RlZi5vcmRlci5sZW5ndGg7IGkrKyl7XG4gICAgICAgICAgICB2YXIgY3VycmVudENvZGUgPSBjb2xvcnNEZWYub3JkZXJbaV07XG4gICAgICAgICAgICBjZWxsLmdyYXBoaWNzLmJlZ2luRmlsbChjZWxsLmNhdGVnb3JpZXNbY3VycmVudENvZGVdLmNvbG9yLnJlcGxhY2UoJyMnLCAnMHgnKSlcbiAgICAgICAgICAgICAgICAuZHJhd1JlY3QoMCwgeSwgdGhpcy5pbnRlcnZhbFdpZHRoLTEsIC1jZWxsLmNhdGVnb3JpZXNbY3VycmVudENvZGVdLmNvdW50ICogaGVpZ2h0U3RlcClcbiAgICAgICAgICAgICAgICAuZW5kRmlsbCgpO1xuICAgICAgICAgICAgeSAtPSBjZWxsLmNhdGVnb3JpZXNbY3VycmVudENvZGVdLmNvdW50KmhlaWdodFN0ZXA7XG4gICAgICAgIH1cbiAgICB9O1xuXG4gICAgdGhpcy5pbml0Q2VsbCA9IGZ1bmN0aW9uKGNlbGwsIGNvbG9yc0RlZil7XG4gICAgICAgIGNlbGwuZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpO1xuICAgICAgICBjZWxsLmdyYXBoaWNzLnBvc2l0aW9uLnggPSB0aGlzLmNpcmNsZVggKyB0aGlzLnJhZGl1cyAqIE1hdGguc2luKGNlbGwuaSooMzYwL3RvdGFsSW5kZXgpKihNYXRoLlBJLzE4MCkpO1xuICAgICAgICBjZWxsLmdyYXBoaWNzLnBvc2l0aW9uLnkgPSB0aGlzLmNpcmNsZVkgLSB0aGlzLnJhZGl1cyAqIE1hdGguY29zKGNlbGwuaSooMzYwL3RvdGFsSW5kZXgpKihNYXRoLlBJLzE4MCkpO1xuICAgICAgICBjZWxsLmdyYXBoaWNzLnJvdGF0aW9uID0gKGNlbGwuaSkqKDM2MC90b3RhbEluZGV4KSooTWF0aC5QSS8xODApICsgKDM2MC8odG90YWxJbmRleCoyKSkqKE1hdGguUEkvMTgwKTtcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY2VsbC5ncmFwaGljcyk7XG5cbiAgICAgICAgZm9yICh2YXIgY2F0ZWdvcnkgaW4gY29sb3JzRGVmLmNvbG9ycyl7XG4gICAgICAgICAgICBjZWxsLmNhdGVnb3JpZXNbY2F0ZWdvcnldID0ge1xuICAgICAgICAgICAgICAgICdjb3VudCc6IDAsXG4gICAgICAgICAgICAgICAgJ2NvbG9yJzogY29sb3JzRGVmLmNvbG9yc1tjYXRlZ29yeV1cbiAgICAgICAgICAgIH07XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHR5cGVvZihjZWxsLmNhdGVnb3JpZXNbJ2RlZmF1bHQnXSkgPT09ICd1bmRlZmluZWQnKXtcbiAgICAgICAgICAgIGNlbGwuY2F0ZWdvcmllc1snZGVmYXVsdCddID0ge1xuICAgICAgICAgICAgICAgICdjb3VudCc6IDAsXG4gICAgICAgICAgICAgICAgJ2NvbG9yJzogY29sb3JzRGVmLmRlZmF1bHRDb2xvclxuICAgICAgICAgICAgfTtcbiAgICAgICAgfVxuICAgIH07XG5cbiAgICB0aGlzLmluaXQgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgaWYgKCF0aGlzLmFyY2hpdmUpe1xuICAgICAgICAgICAgd3MubWVzc2FnZShmdW5jdGlvbihkYXRhKSB7XG4gICAgICAgICAgICAgICAgX3RoaXMuYWRkQW5ub3QoZGF0YSk7XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICh0aGlzLnNob3dDbG9ja0dyYXBoaWNzKXt0aGlzLmluaXRDbG9ja0dyYXBoaWNzKCk7fVxuICAgIH07XG5cblxuICAgIHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgdGhpcy5zdGFydFRzID0gRGF0ZS5ub3coKTtcbiAgICAgICAgbG9hZEFyY2hpdmVzKCk7XG4gICAgfTtcblxuICAgIHRoaXMucmVmcmVzaCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgfTtcblxuICAgIHRoaXMuc3RvcCA9IGZ1bmN0aW9uKCl7XG4gICAgfTtcblxuICAgIHJldHVybiB0aGlzO1xufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBBbm5vdHNUaW1lTGluZTogQW5ub3RzVGltZUxpbmVcbn07XG4iLCIvKipcbioganMvYW5ub3Rzdml6dmlldy5qc1xuKlxuKiBUaGlzIGlzIHRoZSBzdGFydGluZyBwb2ludCBmb3IgeW91ciBhcHBsaWNhdGlvbi5cbiogVGFrZSBhIGxvb2sgYXQgaHR0cDovL2Jyb3dzZXJpZnkub3JnLyBmb3IgbW9yZSBpbmZvXG4qXG4qL1xuLypqc2hpbnQgdW51c2VkOiBmYWxzZSAqL1xuXG4ndXNlIHN0cmljdCc7XG5cbnZhciBQSVhJID0gcmVxdWlyZSgncGl4aScpO1xudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcbnZhciBEb3VibGVSb2xsID0gcmVxdWlyZSgnLi9kb3VibGVyb2xsLmpzJyk7XG52YXIgQW5ub3RzVGltZUxpbmUgPSByZXF1aXJlKCcuL2Fubm90c3RpbWVsaW5lLmpzJyk7XG52YXIgQW5ub3RzUm9sbCA9IHJlcXVpcmUoJy4vYW5ub3Rzcm9sbC5qcycpO1xudmFyIFV0aWxzID0gcmVxdWlyZSgnLi91dGlscy5qcycpO1xuXG52YXIgZGVmYXVsdE9wdGlvbnMgPSB7XG4gICAgeEluaXQ6IDAsXG4gICAgeUluaXQ6IDAsXG4gICAgd2lkdGg6IDEwMjQsXG4gICAgaGVpZ2h0OiA3Njhcbn07XG5cbmZ1bmN0aW9uIEFubm90c1ZpelZpZXcob3B0aW9ucyl7XG4gICAgdmFyIF90aGlzID0gdGhpcztcbiAgICB2YXIgb3B0cyA9IF8ob3B0aW9ucykuZGVmYXVsdHMoZGVmYXVsdE9wdGlvbnMpLnZhbHVlKCk7XG5cbiAgICB0aGlzLmNvbnRhaW5lciA9IG5ldyBQSVhJLkRpc3BsYXlPYmplY3RDb250YWluZXIoKTtcbiAgICB0aGlzLmNvbnRhaW5lci54ID0gb3B0cy54SW5pdDtcbiAgICB0aGlzLmNvbnRhaW5lci55ID0gb3B0cy55SW5pdDtcbiAgICB0aGlzLndpZHRoID0gb3B0cy53aWR0aDtcbiAgICB0aGlzLmhlaWdodD0gb3B0cy5oZWlnaHQ7XG4gICAgdGhpcy50aW1lQmVnaW4gPSBvcHRzLnRpbWVCZWdpbjtcbiAgICB0aGlzLnRpbWVFbmQgPSBvcHRzLnRpbWVFbmQ7XG5cbiAgICB0aGlzLmFubm90Q2F0ZWdvcmllcyA9IFtdO1xuICAgIHRoaXMuYW5ub3RDYXRlZ29yaWVzRGF5MiA9IFtdO1xuICAgIHRoaXMuYW5ub3RDYXRlZ29yaWVzRGF5MSA9IFtdO1xuXG4gICAgdmFyIHdzUGlhbm9yb2xsID0gb3B0cy53c1BpYW5vcm9sbDtcbiAgICB2YXIgd3NBbm5vdCA9IG9wdHMud3NBbm5vdDtcbiAgICB2YXIgc3RhZ2VWaWV3ID0gb3B0cy5zdGFnZVZpZXc7XG4gICAgdmFyIGNoYW5uZWwgPSBvcHRzLmNoYW5uZWw7XG4gICAgdmFyIHNlcnZlclVybCA9IG9wdHMuc2VydmVyVXJsO1xuXG4gICAgdmFyIGV2ZW50Q29kZSA9IG9wdHMuZXZlbnRDb2RlO1xuICAgIHZhciBldmVudENvZGVTZXNzaW9uRGF5MiA9IG9wdHMuZXZlbnRDb2RlU2Vzc2lvbkRheTI7XG5cdHZhciBldmVudENvZGVTZXNzaW9uRGF5MSA9IG9wdHMuZXZlbnRDb2RlU2Vzc2lvbkRheTE7XG4gICAgdmFyIGxvZ2dlciA9IG9wdHMubG9nZ2VyO1xuXG5cdFV0aWxzLmdldEFubm90Q2F0ZWdvcmllcyhvcHRzLnVybENhdGVnb3JpZXMgKyBldmVudENvZGUsIHRoaXMuYW5ub3RDYXRlZ29yaWVzKTtcbiAgICBVdGlscy5nZXRBbm5vdENhdGVnb3JpZXMob3B0cy51cmxDYXRlZ29yaWVzICsgZXZlbnRDb2RlU2Vzc2lvbkRheTIsIHRoaXMuYW5ub3RDYXRlZ29yaWVzRGF5Mik7XG4gICAgVXRpbHMuZ2V0QW5ub3RDYXRlZ29yaWVzKG9wdHMudXJsQ2F0ZWdvcmllcyArIGV2ZW50Q29kZVNlc3Npb25EYXkxLCB0aGlzLmFubm90Q2F0ZWdvcmllc0RheTEpO1xuXG4gICAgc3RhZ2VWaWV3LnJlZ2lzdGVyQ29tcG9uZW50KHRoaXMpO1xuXG4gICAgdmFyIGN1cnJlbnREYXkgPSBuZXcgQW5ub3RzVGltZUxpbmUuQW5ub3RzVGltZUxpbmUoe1xuICAgIFx0c3RhZ2VWaWV3IDogc3RhZ2VWaWV3LFxuICAgICAgICBsb2dnZXI6IGxvZ2dlcixcbiAgICAgICAgd3M6IHdzQW5ub3QsXG4gICAgICAgIHhJbml0OiAwLFxuICAgICAgICB5SW5pdDogMCxcbiAgICAgICAgd2lkdGg6IDEwMjQgLSAyMDAgLSAyMDAsXG4gICAgICAgIGhlaWdodDogNzY4LTIwMCxcbiAgICAgICAgdGltZUJlZ2luOiB0aGlzLnRpbWVCZWdpbixcbiAgICAgICAgdGltZUVuZDogdGhpcy50aW1lRW5kLFxuICAgICAgICBpbnRlcnZhbFdpZHRoOiA2LFxuICAgICAgICBpbnRlcnZhbEhlaWdodDogMTAsXG4gICAgICAgIG1heENlbGxIZWlnaHQ6IDcwLFxuICAgICAgICByYWRpdXM6IDIwMCxcbiAgICAgICAgZXZlbnRDb2RlOiBldmVudENvZGUsXG4gICAgICAgIGNoYW5uZWw6IGNoYW5uZWwsXG4gICAgICAgIHNlcnZlclVybDogc2VydmVyVXJsLFxuICAgICAgICBhbm5vdENhdGVnb3JpZXM6IHRoaXMuYW5ub3RDYXRlZ29yaWVzXG4gICAgfSk7XG5cblx0Ly9BcmNoaXZlIGRheSAyXG4gICAgdmFyIHRpbWVMaW5lRGF5MiA9IG5ldyBBbm5vdHNUaW1lTGluZS5Bbm5vdHNUaW1lTGluZSh7XG4gICAgXHRzdGFnZVZpZXcgOiBzdGFnZVZpZXcsXG4gICAgICAgIHhJbml0OiAwLFxuICAgICAgICB5SW5pdDogMCxcbiAgICAgICAgd2lkdGg6IDEwMjQgLSAyMDAgLSAyMDAsXG4gICAgICAgIGhlaWdodDogNzY4LTIwMCxcbiAgICAgICAgdGltZUJlZ2luOiBEYXRlLnBhcnNlKCcyMDE1LTAxLTIzVDA5OjMwOjAwKzAxOjAwJyksXG4gICAgICAgIHRpbWVFbmQ6IERhdGUucGFyc2UoJzIwMTUtMDEtMjNUMTg6MzA6MDArMDE6MDAnKSxcbiAgICAgICAgY2lyY2xlWDogY3VycmVudERheS5jaXJjbGVYLFxuICAgICAgICBjaXJjbGVZOiBjdXJyZW50RGF5LmNpcmNsZVksXG4gICAgICAgIGludGVydmFsV2lkdGg6IChjdXJyZW50RGF5LnJhZGl1cyoyLzMpKiBjdXJyZW50RGF5LmludGVydmFsV2lkdGggLyBjdXJyZW50RGF5LnJhZGl1cyxcbiAgICAgICAgaW50ZXJ2YWxIZWlnaHQ6IChjdXJyZW50RGF5LmludGVydmFsSGVpZ2h0ICogKGN1cnJlbnREYXkucmFkaXVzIC0gY3VycmVudERheS5yYWRpdXMqMi8zKSkvIGN1cnJlbnREYXkubWF4Q2VsbEhlaWdodCxcbiAgICAgICAgbWF4Q2VsbEhlaWdodDogKGN1cnJlbnREYXkucmFkaXVzIC0gY3VycmVudERheS5yYWRpdXMqMi8zKS8yLFxuICAgICAgICByYWRpdXM6IGN1cnJlbnREYXkucmFkaXVzKjIvMyxcbiAgICAgICAgYW5ub3RDYXRlZ29yaWVzOiB0aGlzLmFubm90Q2F0ZWdvcmllc0RheTIsXG4gICAgICAgIGV2ZW50Q29kZTogZXZlbnRDb2RlU2Vzc2lvbkRheTIsXG4gICAgICAgIGNoYW5uZWw6IGNoYW5uZWwsXG4gICAgICAgIHNlcnZlclVybDogc2VydmVyVXJsLFxuICAgICAgICBzaG93Q2xvY2tHcmFwaGljczpmYWxzZSxcbiAgICAgICAgYXJjaGl2ZTogdHJ1ZVxuICAgIH0pO1xuXG5cdC8vQXJjaGl2ZSBkYXkgMVxuICAgIHZhciB0aW1lTGluZURheTEgPSBuZXcgQW5ub3RzVGltZUxpbmUuQW5ub3RzVGltZUxpbmUoe1xuICAgIFx0c3RhZ2VWaWV3IDogc3RhZ2VWaWV3LFxuICAgICAgICB4SW5pdDogMCxcbiAgICAgICAgeUluaXQ6IDAsXG4gICAgICAgIHdpZHRoOiAxMDI0IC0gMjAwIC0gMjAwLFxuICAgICAgICBoZWlnaHQ6IDc2OC0yMDAsXG4gICAgICAgIHRpbWVCZWdpbjogRGF0ZS5wYXJzZSgnMjAxNS0wMS0yMlQwOTozMDowMCswMTowMCcpLFxuICAgICAgICB0aW1lRW5kOiBEYXRlLnBhcnNlKCcyMDE1LTAxLTIyVDE4OjMwOjAwKzAxOjAwJyksXG4gICAgICAgIGNpcmNsZVg6IGN1cnJlbnREYXkuY2lyY2xlWCxcbiAgICAgICAgY2lyY2xlWTogY3VycmVudERheS5jaXJjbGVZLFxuICAgICAgICBpbnRlcnZhbFdpZHRoOiAoY3VycmVudERheS5yYWRpdXMqMi8zKSogY3VycmVudERheS5pbnRlcnZhbFdpZHRoIC8gY3VycmVudERheS5yYWRpdXMsXG4gICAgICAgIGludGVydmFsSGVpZ2h0OiAoY3VycmVudERheS5pbnRlcnZhbEhlaWdodCAqIChjdXJyZW50RGF5LnJhZGl1cyAtIGN1cnJlbnREYXkucmFkaXVzLzMpKS8gY3VycmVudERheS5tYXhDZWxsSGVpZ2h0LFxuICAgICAgICBtYXhDZWxsSGVpZ2h0OiAoY3VycmVudERheS5yYWRpdXMqMi8zIC0gY3VycmVudERheS5yYWRpdXMvMykvNCxcbiAgICAgICAgcmFkaXVzOiBjdXJyZW50RGF5LnJhZGl1cy8zLFxuICAgICAgICBhbm5vdENhdGVnb3JpZXM6IHRoaXMuYW5ub3RDYXRlZ29yaWVzRGF5MSxcbiAgICAgICAgZXZlbnRDb2RlOiBldmVudENvZGVTZXNzaW9uRGF5MSxcbiAgICAgICAgY2hhbm5lbDogY2hhbm5lbCxcbiAgICAgICAgc2VydmVyVXJsOiBzZXJ2ZXJVcmwsXG4gICAgICAgIHNob3dDbG9ja0dyYXBoaWNzOmZhbHNlLFxuICAgICAgICBhcmNoaXZlOiB0cnVlLFxuICAgIH0pO1xuXG4gICAgdmFyIGN1cnJlbnRUaW1lVGV4dCA9IG5ldyBQSVhJLlRleHQoJy0tIDogLS0gOiAtLScsIHsgZm9udDogJzE4cHQgR290aGljIFN0YW5kYXJkJywgZmlsbDogJyM2NDY0NjQnIH0pO1xuICAgIGN1cnJlbnRUaW1lVGV4dC54ID0gdGltZUxpbmVEYXkyLmNpcmNsZVggLSBjdXJyZW50VGltZVRleHQud2lkdGgvMjtcbiAgICBjdXJyZW50VGltZVRleHQueSA9IHRpbWVMaW5lRGF5Mi5jaXJjbGVZIC0gY3VycmVudFRpbWVUZXh0LmhlaWdodC8yO1xuICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGN1cnJlbnRUaW1lVGV4dCk7XG5cblxuICAgIHZhciBkb3VibGVSb2xsSCA9IG5ldyBEb3VibGVSb2xsLkRvdWJsZVJvbGwoe1xuICAgICAgICBzdGFnZVZpZXcgOiBzdGFnZVZpZXcsXG4gICAgICAgIGxvZ2dlcjogbG9nZ2VyLFxuICAgICAgICB3czogd3NQaWFub3JvbGwsXG4gICAgICAgIHlJbml0OiAodGhpcy5oZWlnaHQgLSAyMDApLFxuICAgICAgICBzY2VuZUhlaWdodDogMjAwLFxuICAgICAgICBwaWFub3JvbGxzIDogW1xuICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgIGhlaWdodDogMjAwLFxuICAgICAgICAgICAgICAgIHRpbWVXaWR0aDogMTAsXG4gICAgICAgICAgICAgICAgbGluZUludGVydmFsOiA1MDAwLFxuICAgICAgICAgICAgICAgIG5vdGVIZWlnaHQ6IDEwXG4gICAgICAgICAgICB9LFxuICAgICAgICBdXG4gICAgfSk7XG5cbiAgICB2YXIgZG91YmxlUm9sbFYgPSBuZXcgRG91YmxlUm9sbC5Eb3VibGVSb2xsKHtcbiAgICAgICAgc3RhZ2VWaWV3IDogc3RhZ2VWaWV3LFxuICAgICAgICBsb2dnZXI6IGxvZ2dlcixcbiAgICAgICAgd3M6IHdzUGlhbm9yb2xsLFxuICAgICAgICBvcmllbnRhdGlvbjogJ3ZlcnRpY2FsJyxcbiAgICAgICAgc2NlbmVIZWlnaHQ6IDc2OC0yMDAsXG4gICAgICAgIHBpYW5vcm9sbHMgOiBbXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgaGVpZ2h0OiAyMDAsXG4gICAgICAgICAgICAgICAgdGltZVdpZHRoOiA2MCxcbiAgICAgICAgICAgICAgICBsaW5lSW50ZXJ2YWw6IDUwMDAsXG4gICAgICAgICAgICAgICAgbm90ZUhlaWdodDogNSxcbiAgICAgICAgICAgIH0sXG4gICAgICAgIF1cbiAgICB9KTtcblxuICAgIHZhciBhbm5vdHNSb2xsID0gbmV3IEFubm90c1JvbGwuQW5ub3RzUm9sbCh7XG4gICAgICAgIHN0YWdlVmlldyA6IHN0YWdlVmlldyxcbiAgICAgICAgbG9nZ2VyOiBsb2dnZXIsXG4gICAgICAgIHdzOiB3c0Fubm90LFxuICAgICAgICBwYXJlbnRDb250YWluZXI6IGRvdWJsZVJvbGxWLnN0YWdlLFxuICAgICAgICB4SW5pdDogMTAyNCAtIDIwMCAtIDIwMCxcbiAgICAgICAgeUluaXQ6IDc2OC0yMDAsXG4gICAgICAgIHdpZHRoOiAyMDAgKyAyMDAsXG4gICAgICAgIGhlaWdodDogNzY4LTIwMCxcbiAgICAgICAgd2lkdGhSb2xsOiAyMDAsXG4gICAgICAgIGZyYW1lcmF0ZTogZG91YmxlUm9sbFYuZnJhbWVyYXRlLFxuICAgICAgICBwaXhlbHNQZXJTZWNvbmQ6IE1hdGguZmxvb3IoMTAyNCAvIDYwKSxcbiAgICAgICAgYW5ub3RDb2xvcnM6IHRoaXMuYW5ub3RDYXRlZ29yaWVzXG4gICAgfSk7XG5cbiAgICB2YXIgbGltaXRlcnMgPSBuZXcgUElYSS5HcmFwaGljcygpXG4gICAgICAgIC5saW5lU3R5bGUoMSwgMHg2NDY0NjQpXG4gICAgICAgIC5tb3ZlVG8oYW5ub3RzUm9sbC5jb250YWluZXIueCwgYW5ub3RzUm9sbC5jb250YWluZXIueSlcbiAgICAgICAgLmxpbmVUbyhhbm5vdHNSb2xsLmNvbnRhaW5lci54LCBhbm5vdHNSb2xsLmNvbnRhaW5lci55IC0gYW5ub3RzUm9sbC5oZWlnaHQpXG4gICAgICAgIC5tb3ZlVG8oYW5ub3RzUm9sbC5jb250YWluZXIueCArIGFubm90c1JvbGwud2lkdGhSb2xsLCBhbm5vdHNSb2xsLmNvbnRhaW5lci55KVxuICAgICAgICAubGluZVRvKGFubm90c1JvbGwuY29udGFpbmVyLnggKyBhbm5vdHNSb2xsLndpZHRoUm9sbCwgYW5ub3RzUm9sbC5jb250YWluZXIueSAtIGFubm90c1JvbGwuaGVpZ2h0KVxuICAgICAgICAubW92ZVRvKDAsIHRoaXMuaGVpZ2h0IC0gMjAwKVxuICAgICAgICAubGluZVRvKHRoaXMud2lkdGgsIHRoaXMuaGVpZ2h0IC0gMjAwKVxuICAgICAgICAuZHJhd1JlY3QoMCwgMCwgdGhpcy53aWR0aCAtMSwgdGhpcy5oZWlnaHQgLTEpXG4gICAgICAgIC5iZWdpbkZpbGwoMHhFQ0VDRUMpXG4gICAgICAgIC5kcmF3UmVjdCgxMDI0IC0gMjAwLCAwLCAyMDAsIDc2OC0yMDApXG4gICAgICAgIC5lbmRGaWxsKCk7XG4gICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQobGltaXRlcnMpO1xuXG4gICAgdGhpcy5pbml0ID0gZnVuY3Rpb24oKXtcbiAgICB9O1xuXG4gICAgdGhpcy51cGRhdGVUaW1lID0gZnVuY3Rpb24oKXtcbiAgICAgICAgY3VycmVudFRpbWVUZXh0LnNldFRleHQoVXRpbHMuZm9ybWF0VGltZShEYXRlLm5vdygpKSk7XG4gICAgfTtcblxuICAgIHZhciByZWZyZXNoVGltZUludGVydmFsO1xuXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICByZWZyZXNoVGltZUludGVydmFsID0gc2V0SW50ZXJ2YWwoZnVuY3Rpb24oKSB7X3RoaXMudXBkYXRlVGltZSgpO30sIDEwMDApO1xuICAgIH07XG5cbiAgICB0aGlzLnJlZnJlc2ggPSBmdW5jdGlvbigpIHtcbiAgICB9O1xuXG4gICAgdGhpcy5zdG9wID0gZnVuY3Rpb24oKXtcbiAgICB9O1xuXG4gICAgcmV0dXJuIHRoaXM7XG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgQW5ub3RzVml6VmlldzogQW5ub3RzVml6Vmlld1xufTtcbiIsIi8qKlxuKiBzY3JpcHRzL2RvdWJsZXJvbGwuanNcbipcbiogVGhpcyBpcyB0aGUgc3RhcnRpbmcgcG9pbnQgZm9yIHlvdXIgYXBwbGljYXRpb24uXG4qIFRha2UgYSBsb29rIGF0IGh0dHA6Ly9icm93c2VyaWZ5Lm9yZy8gZm9yIG1vcmUgaW5mb1xuKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG5cbnZhciBQSVhJID0gcmVxdWlyZSgncGl4aScpO1xudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcbnZhciBQaWFub1JvbGwgPSByZXF1aXJlKCcuL3BpYW5vcm9sbCcpO1xuXG52YXIgZGVmYXVsdENvbmZpZyA9IHtcbiAgICBvcmllbnRhdGlvbjogJ2hvcml6b250YWwnLFxuICAgIGxvZ2dlcjogdW5kZWZpbmVkLFxuICAgIHNjZW5lV2lkdGg6IDEwMjQsXG4gICAgcGlhbm9yb2xscyA6IFtcbiAgICAgIHtcbiAgICAgICAgaGVpZ2h0OiAzODQsXG4gICAgICAgIHRpbWVXaWR0aDogMTAsXG4gICAgICAgIGxpbmVJbnRlcnZhbDogNTAwMCxcbiAgICAgICAgbm90ZUhlaWdodDogdW5kZWZpbmVkLFxuICAgICAgICByYW5nZToge1xuICAgIFx0XHRib3R0b206IDQwLFxuICAgIFx0XHR0b3A6IDkwLFxuICAgICAgICB9LFxuICAgICAgICBkeW5hbWljUmFuZ2U6IHRydWUsXG4gICAgICB9LFxuICAgICAge1xuICAgICAgICBoZWlnaHQ6IDM4NCxcbiAgICAgICAgdGltZVdpZHRoOiA2MCxcbiAgICAgICAgbGluZUludGVydmFsOiA1MDAwLFxuICAgICAgICBub3RlSGVpZ2h0OiB1bmRlZmluZWQsXG4gICAgICAgIHJhbmdlOntcbiAgICBcdFx0Ym90dG9tOiAwLFxuICAgIFx0XHR0b3A6IDEyOCxcbiAgICAgICAgfSxcbiAgICAgICAgZHluYW1pY1JhbmdlOiBmYWxzZSxcbiAgICAgIH0sXG4gICAgXSxcbiAgICBmcmFtZXJhdGU6IDI1LFxuICAgIG9mZnNldE11c2ljOiBmYWxzZSxcbiAgICBzY2VuZUJnQ29sb3I6IDB4RkZGRkZGLFxuICAgIGxpbmVDb2xvcjogMHg0NDQ0NDQsXG4gICAgbGluZUZpbGxDb2xvcjogMHhGRkZGMDAsXG4gICAgbm90ZUNvbG9yczogWzB4QjkwMDAwLCAweDRCREQ3MSwgMHhBRjkzMUUsIDB4MUMyOEJBLCAweDUzNjk5MV0sXG4gICAgbm90ZUhlaWdodDogdW5kZWZpbmVkLFxuICAgIHplcm9TaGlmdDogMC45LFxuICAgIHRpbWVXaWR0aDogNjAsXG4gICAgbGluZUludGVydmFsOiA1MDAwLFxuLy8gICAgd3NVcmk6IHVuZGVmaW5lZCxcbi8vICAgIGV2ZW50Q29kZTogdW5kZWZpbmVkXG5cbn07XG5cbmZ1bmN0aW9uIERvdWJsZVJvbGwob3B0aW9ucykge1xuXG4gICAgdmFyIF90aGlzID0gdGhpcztcbiAgICB2YXIgb3B0cyA9IF8ob3B0aW9ucykuZGVmYXVsdHMoZGVmYXVsdENvbmZpZykudmFsdWUoKTtcblxuICAgIHZhciBvcmllbnRhdGlvbiA9IG9wdHMub3JpZW50YXRpb247XG4gICAgdmFyIGlzSG9yaXpvbnRhbCA9IChvcmllbnRhdGlvbiAhPT0gJ3ZlcnRpY2FsJyk7XG5cbiAgICB0aGlzLmxvZ2dlciA9IG9wdHMubG9nZ2VyO1xuICAgIHRoaXMubGluZUNvbG9yID0gb3B0cy5saW5lQ29sb3I7XG4gICAgdGhpcy5saW5lRmlsbENvbG9yID0gb3B0cy5saW5lRmlsbENvbG9yO1xuICAgIHRoaXMuZnJhbWVyYXRlID0gb3B0cy5mcmFtZXJhdGU7XG4gICAgdGhpcy5vZmZzZXRNdXNpYyA9IG9wdHMub2Zmc2V0TXVzaWM7XG4gICAgdGhpcy5ub3RlQ29sb3JzID0gb3B0cy5ub3RlQ29sb3JzO1xuXG4gICAgdmFyIG5vdGVIZWlnaHQgPSBvcHRzLm5vdGVIZWlnaHQ7XG4gICAgdmFyIHNjZW5lSGVpZ2h0ID0gb3B0cy5zY2VuZUhlaWdodCB8fCBfKG9wdHMucGlhbm9yb2xscykucmVkdWNlKGZ1bmN0aW9uKHMscCkgeyByZXR1cm4gcyArIHAuaGVpZ2h0OyB9LCAwKTtcbiAgICB2YXIgdGltZVdpZHRoID0gb3B0cy50aW1lV2lkdGg7XG4gICAgdmFyIGxpbmVJbnRlcnZhbCA9IG9wdHMubGluZUludGVydmFsO1xuICAgIHZhciBvZmZzZXRNdXNpYyA9IG9wdHMub2Zmc2V0TXVzaWM7XG5cbiAgICB2YXIgc2NlbmVXaWR0aCA9IG9wdHMuc2NlbmVXaWR0aDtcbiAgICB2YXIgc3RhZ2VWaWV3ID0gb3B0cy5zdGFnZVZpZXc7XG5cbiAgICB2YXIgemVyb1NoaWZ0ID0gb3B0cy56ZXJvU2hpZnQ7XG5cbiAgICB2YXIgd3MgPSBvcHRzLndzO1xuXG4gICAgdmFyIGNvbG9yc1JlZyA9IHt9O1xuXG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IE1hdGguZmxvb3Ioc2NlbmVXaWR0aCp6ZXJvU2hpZnQpO1xuICAgIHRoaXMuY29udGFpbmVyLnkgPSAwO1xuXG4gICAgc3RhZ2VWaWV3LnJlZ2lzdGVyQ29tcG9uZW50KHRoaXMpO1xuXG4gICAgdmFyIHBpYW5vcm9sbExpc3QgPSBbXTtcblxuICAgIHZhciBwaWFub3JvbGxPcHRpb25zID0ge1xuICAgICAgICBwYXJlbnRDb250YWluZXI6IHRoaXMuY29udGFpbmVyLFxuICAgICAgICBvcmllbnRhdGlvbjogb3JpZW50YXRpb24sXG4gICAgICAgIHhJbml0OiAwLFxuICAgICAgICB3aWR0aDogc2NlbmVXaWR0aCxcbiAgICAgICAgbm90ZUNvbG9yczogdGhpcy5ub3RlQ29sb3JzLFxuICAgICAgICBjb2xvcnNSZWc6IGNvbG9yc1JlZyxcbiAgICAgICAgbGluZUNvbG9yOiB0aGlzLmxpbmVDb2xvcixcbiAgICAgICAgbGluZUludGVydmFsOiBsaW5lSW50ZXJ2YWwsXG4gICAgICAgIG9mZnNldE11c2ljOiBvZmZzZXRNdXNpYyxcbiAgICB9O1xuXG4gICAgdmFyIHlJbml0ID0gb3B0cy55SW5pdCB8fCAwO1xuICAgIHZhciBsaW5lc0Rvd24gPSB0cnVlO1xuICAgIF8ob3B0cy5waWFub3JvbGxzKS5mb3JFYWNoKGZ1bmN0aW9uKHByRGVmLCBpKSB7XG4gICAgICAgIHZhciBwck5vdGVIZWlnaHQgPSBub3RlSGVpZ2h0IHx8IHByRGVmLm5vdGVIZWlnaHQgfHwgcHJEZWYuaGVpZ2h0IC8gKHByRGVmLnJhbmdlLnRvcCAtIHByRGVmLnJhbmdlLmJvdHRvbSArIDEpO1xuICAgICAgICB2YXIgcHJUaW1lV2lkdGggPSBwckRlZi50aW1lV2lkdGggfHwgdGltZVdpZHRoO1xuICAgICAgICBwaWFub3JvbGxMaXN0LnB1c2gobmV3IFBpYW5vUm9sbChfKHtcbiAgICAgICAgICAgIHlJbml0OiB5SW5pdCxcbiAgICAgICAgICAgIGhlaWdodDogcHJEZWYuaGVpZ2h0LFxuICAgICAgICAgICAgbGluZXNEb3duOiBsaW5lc0Rvd24sXG4gICAgICAgICAgICBwaXhlbHNQZXJTZWNvbmQ6IE1hdGguZmxvb3Ioc2NlbmVXaWR0aCAvIHByVGltZVdpZHRoKSxcbiAgICAgICAgICAgIG5vdGVIZWlnaHQ6IHByTm90ZUhlaWdodCxcbiAgICAgICAgICAgIGxpbmVJbnRlcnZhbDogcHJEZWYubGluZUludGVydmFsLFxuICAgICAgICAgICAgcmFuZ2U6IHByRGVmLnJhbmdlLFxuICAgICAgICAgICAgZHluYW1pY1JhbmdlOiBwckRlZi5keW5hbWljUmFuZ2UsXG4gICAgICAgIH0pLmRlZmF1bHRzKHBpYW5vcm9sbE9wdGlvbnMpLnZhbHVlKCkpKTtcbiAgICAgICAgeUluaXQgKz0gcHJEZWYuaGVpZ2h0O1xuICAgICAgICBsaW5lc0Rvd24gPSAhbGluZXNEb3duO1xuXG4gICAgICAgIGlmKGk8KG9wdHMucGlhbm9yb2xscy5sZW5ndGgtMSkpIHtcbiAgICAgICAgICAgIHZhciBsaW5lR3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpXG4gICAgICAgICAgICAgICAgLmJlZ2luRmlsbChfdGhpcy5saW5lRmlsbENvbG9yKVxuICAgICAgICAgICAgICAgIC5saW5lU3R5bGUoMSwgX3RoaXMubGluZUNvbG9yKVxuICAgICAgICAgICAgICAgIC5tb3ZlVG8oTWF0aC5mbG9vcihzY2VuZVdpZHRoKnplcm9TaGlmdCksIHlJbml0KVxuICAgICAgICAgICAgICAgIC5saW5lVG8oLXNjZW5lV2lkdGggLSBNYXRoLmZsb29yKHNjZW5lV2lkdGgqemVyb1NoaWZ0KSwgeUluaXQpXG4gICAgICAgICAgICAgICAgLmVuZEZpbGwoKTtcbiAgICAgICAgICAgIF90aGlzLmNvbnRhaW5lci5hZGRDaGlsZChsaW5lR3JhcGhpY3MpO1xuICAgICAgICB9XG4gICAgfSk7XG5cbiAgICBpZighaXNIb3Jpem9udGFsKSB7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLnJvdGF0aW9uID0gTWF0aC5QSS8yO1xuICAgICAgICB0aGlzLmNvbnRhaW5lci55ID0gc2NlbmVIZWlnaHQ7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLnggPSBzY2VuZVdpZHRoO1xuICAgIH1cblxuXG4gICAgdGhpcy5pbml0ID0gZnVuY3Rpb24oKSB7XG5cbiAgICBcdHdzLm1lc3NhZ2UoZnVuY3Rpb24oZGF0YSkge1xuICAgICAgICAgICAgX3RoaXMuYWRkTm90ZXMoZGF0YSk7XG4gICAgICAgIH0pO1xuXG4gICAgfTtcblxuXG4gICAgdGhpcy5hZGROb3RlcyA9IGZ1bmN0aW9uKGRhdGEpIHtcblxuICAgICAgICBwaWFub3JvbGxMaXN0LmZvckVhY2goZnVuY3Rpb24oYykge1xuICAgICAgICAgICAgYy5hZGROb3RlUmF3KGRhdGEpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIHBpYW5vcm9sbExpc3QuZm9yRWFjaChmdW5jdGlvbihjKSB7XG4gICAgICAgICAgICBjLm1vdmUoKTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIC8vIEluaXQgcGFnZSBhbmQgaW50ZXJ2YWxzXG4gICAgdmFyIHN0YXJ0VHM7XG5cbiAgICB0aGlzLnN0YXJ0ID0gZnVuY3Rpb24oKSB7XG5cbiAgICAgICAgc3RhcnRUcyA9IERhdGUubm93KCk7XG4gICAgICAgIHBpYW5vcm9sbExpc3QuZm9yRWFjaChmdW5jdGlvbihjKSB7XG4gICAgICAgICAgICBjLnN0YXJ0KCk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cbiAgICB0aGlzLnN0b3AgPSBmdW5jdGlvbigpIHtcblxuICAgICAgICBwaWFub3JvbGxMaXN0LmZvckVhY2goZnVuY3Rpb24oYykge1xuICAgICAgICAgICAgYy5zdG9wKCk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cblxuICAgIHRoaXMubG9nID0gZnVuY3Rpb24obSkge1xuICAgICAgICBpZih0aGlzLmxvZ2dlcikge1xuICAgICAgICAgICAgdGhpcy5sb2dnZXIubG9nKG0pO1xuICAgICAgICB9XG4gICAgfTtcblxuXG5cbiAgICByZXR1cm4gdGhpcztcbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgRG91YmxlUm9sbDogRG91YmxlUm9sbFxufTtcbiIsIi8qKlxuKiBqcy93c3dyYXBwZXIuanNcbipcbiogc2ltcGxlIGxvZ2dlciBzZXJ2aWNlXG4qXG4qL1xuXG4vKiBnbG9iYWwgZG9jdW1lbnQ6IGZhbHNlICovXG5cbid1c2Ugc3RyaWN0JztcblxuZnVuY3Rpb24gSHRtbExvZ2dlcihkb0xvZywgY29udGFpbmVyKSB7XG5cbiAgICB2YXIgbG9nQ29udGFpbmVyID0gY29udGFpbmVyO1xuICAgIGlmKHR5cGVvZihjb250YWluZXIpID09PSAnc3RyaW5nJykge1xuICAgICAgICBsb2dDb250YWluZXIgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChjb250YWluZXIpO1xuICAgIH1cbiAgICBpZighZG9Mb2cpIHtcbiAgICAgICAgZG9jdW1lbnQuYm9keS5yZW1vdmVDaGlsZChsb2dDb250YWluZXIpO1xuICAgICAgICBsb2dDb250YWluZXIgPSB1bmRlZmluZWQ7XG4gICAgfVxuXG5cbiAgICB0aGlzLmxvZyA9IGZ1bmN0aW9uKG1zZykge1xuICAgICAgICBpZihkb0xvZyAmJiBsb2dDb250YWluZXIpIHtcbiAgICAgICAgICAgIGxvZ0NvbnRhaW5lci5pbm5lckhUTUwgKz0gbXNnICsgJ1xcbic7XG4gICAgICAgICAgICBsb2dDb250YWluZXIuc2Nyb2xsVG9wID0gbG9nQ29udGFpbmVyLnNjcm9sbEhlaWdodDtcbiAgICAgICAgfVxuICAgIH07XG59XG5cbmZ1bmN0aW9uIENvbnNvbGVMb2dnZXIoZG9Mb2cpIHtcblxuICAgIHRoaXMubG9nID0gZnVuY3Rpb24obXNnKSB7XG4gICAgICAgIGlmKGRvTG9nKSB7XG4gICAgICAgICAgICBjb25zb2xlLmxvZyhtc2cpO1xuICAgICAgICB9XG4gICAgfTtcblxufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBIdG1sTG9nZ2VyOiBIdG1sTG9nZ2VyLFxuICAgIENvbnNvbGVMb2dnZXI6IENvbnNvbGVMb2dnZXJcbn07XG4iLCIvKipcbioganMvcGlhbm9yb2xsLmpzXG4qXG4qIHBpYW5vcm9sbCBiYXNpYyBjb21wb25lbnRcbipcbiovXG5cbid1c2Ugc3RyaWN0JztcblxuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciByYW5kb21Db2xvciA9IHJlcXVpcmUoJ3JhbmRvbUNvbG9yJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xuXG52YXIgTlRQX0VQT0NIX0RFTFRBID0gMjIwODk4ODgwMDsgLy9jLmYuIFJGQyA4NjhcblxuZnVuY3Rpb24gUGlhbm9Sb2xsKG9wdGlvbnMpIHtcbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHRoaXMuY29udGFpbmVyID0gbmV3IFBJWEkuRGlzcGxheU9iamVjdENvbnRhaW5lcigpO1xuICAgIHRoaXMuY29udGFpbmVyLnggPSBvcHRpb25zLnhJbml0O1xuICAgIHRoaXMuY29udGFpbmVyLnkgPSBvcHRpb25zLnlJbml0O1xuICAgIG9wdGlvbnMucGFyZW50Q29udGFpbmVyLmFkZENoaWxkKHRoaXMuY29udGFpbmVyKTtcblxuICAgIHZhciBvcmllbnRhdGlvbiA9IG9wdGlvbnMub3JpZW50YXRpb247XG4gICAgdmFyIGlzSG9yaXpvbnRhbCA9IChvcmllbnRhdGlvbiAhPT0gJ3ZlcnRpY2FsJyk7XG5cbiAgICB0aGlzLmxpbmVzRG93biA9IG9wdGlvbnMubGluZXNEb3duO1xuICAgIHRoaXMuaGVpZ2h0ID0gb3B0aW9ucy5oZWlnaHQ7XG4gICAgdGhpcy5waXhlbHNQZXJTZWNvbmQgPSBvcHRpb25zLnBpeGVsc1BlclNlY29uZDtcbiAgICB0aGlzLndpZHRoID0gb3B0aW9ucy53aWR0aDtcbiAgICB0aGlzLm5vdGVDb2xvcnMgPSBvcHRpb25zLm5vdGVDb2xvcnM7XG4gICAgdGhpcy5jb2xvcnNSZWcgPSBvcHRpb25zLmNvbG9yc1JlZyB8fCB7fTtcbiAgICB0aGlzLmxpbmVDb2xvciA9IG9wdGlvbnMubGluZUNvbG9yO1xuICAgIHRoaXMubGluZUludGVydmFsID0gb3B0aW9ucy5saW5lSW50ZXJ2YWw7XG4gICAgdGhpcy5vZmZzZXRNdXNpYyA9IG9wdGlvbnMub2Zmc2V0TXVzaWMgfHwgZmFsc2U7XG4gICAgdGhpcy5ub3RlSGVpZ2h0ID0gb3B0aW9ucy5ub3RlSGVpZ2h0O1xuICAgIHRoaXMubm90ZURpY3QgPSB7fTtcbiAgICB0aGlzLnN0YXJ0VHMgPSBvcHRpb25zLnN0YXJ0VHMgfHwgRGF0ZS5ub3coKTtcbiAgICB0aGlzLmR5bmFtaWNSYW5nZSA9IG9wdGlvbnMuZHluYW1pY1JhbmdlO1xuICAgIHRoaXMuaW5pdFJhbmdlID0gb3B0aW9ucy5yYW5nZTtcbiAgICB0aGlzLnJhbmdlID0gb3B0aW9ucy5yYW5nZTtcbiAgICBcblxuICAgIHZhciBzdGFydGVkID0gZmFsc2U7XG5cbiAgICB2YXIgaXNIaWRkZW4gPSBmdW5jdGlvbihjaGlsZCkge1xuICAgICAgICAvLyBUT0RPOiB0aGUgb3JpZ2luIHBvaW50IGlzIGFuIGFwcHJveGltYXRpb24uIFNob3VsZCByZWZpbmUgdGhpc1xuICAgICAgICB2YXIgZ2xvYmFsUG9zID0gY2hpbGQudG9HbG9iYWwobmV3IFBJWEkuUG9pbnQoMCwwKSk7XG4gICAgICAgIHJldHVybiAoKGdsb2JhbFBvcy54ICsgY2hpbGQud2lkdGgpIDwgMCkgfHwgKChnbG9iYWxQb3MueSArIGNoaWxkLmhlaWdodCkgPCAwKSA7XG4gICAgfTtcblxuICAgIC8vVE9ETzogSSBkbyBub3QgbGlrZSB0aGUgXCJyZWdDb2xvclwiIG9iamVjdC4gVGhpcyBzaG91bGQgbm90IGJlIGdsb2JhbCwgYnV0IGxvY2FsXG4gICAgdGhpcy5nZXRDb2xvciA9IGZ1bmN0aW9uKGNhbmFsKSB7XG4gICAgICAgIHZhciBjb2xvciA9IHRoaXMuY29sb3JzUmVnW2NhbmFsXTtcbiAgICAgICAgaWYodHlwZW9mKGNvbG9yKSA9PT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgICAgICAgIHZhciBjb2xvcnNSZWdTaXplID0gT2JqZWN0LmtleXModGhpcy5jb2xvcnNSZWcpLmxlbmd0aDtcbiAgICAgICAgICAgIGlmKGNvbG9yc1JlZ1NpemUgPCB0aGlzLm5vdGVDb2xvcnMubGVuZ3RoKSB7XG4gICAgICAgICAgICAgICAgY29sb3IgPSB0aGlzLmNvbG9yc1JlZ1tjYW5hbF0gPSB0aGlzLm5vdGVDb2xvcnNbY29sb3JzUmVnU2l6ZV07XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBjb2xvciA9IHRoaXMuY29sb3JzUmVnW2NhbmFsXSA9IHBhcnNlSW50KHJhbmRvbUNvbG9yKHsgbHVtaW5vc2l0eTogJ2xpZ2h0JywgaHVlOiAncmFuZG9tJywgZm9ybWF0OidoZXgnfSkucmVwbGFjZSgvXiMvLCAnJyksIDE2KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gY29sb3I7XG4gICAgfTtcblxuICAgIHRoaXMuZ2V0Tm90ZVJlY3QgPSBmdW5jdGlvbihub3RlLCB4LCB5LCBjb2xvciwgYWxwaGEsIHdpZHRoLCBoZWlnaHQpIHtcbiAgICAgICAgdmFyIGdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKTtcbiAgICAgICAgZ3JhcGhpY3Mubm90ZSA9IG5vdGU7XG4gICAgICAgIGdyYXBoaWNzLmJlZ2luRmlsbChjb2xvciwgYWxwaGEpO1xuICAgICAgICBncmFwaGljcy5kcmF3UmVjdCgwLCAwLCB3aWR0aCwgaGVpZ2h0KTtcbiAgICAgICAgZ3JhcGhpY3MuZW5kRmlsbCgpO1xuICAgICAgICBncmFwaGljcy54ID0geDtcbiAgICAgICAgZ3JhcGhpY3MueSA9IHk7XG4gICAgICAgIGdyYXBoaWNzLndpZHRoID0gd2lkdGg7XG4gICAgICAgIGdyYXBoaWNzLmhlaWdodCA9IGhlaWdodDtcbiAgICAgICAgcmV0dXJuIGdyYXBoaWNzO1xuICAgIH07XG5cbiAgICB0aGlzLmFkZE5vdGVSYXcgPSBmdW5jdGlvbihkYXRhKSB7XG4gICAgICAgIHZhciBub3RlID0gZGF0YS5jb250ZW50WzNdO1xuICAgICAgICB2YXIgdmVsb2NpdHkgPSBkYXRhLmNvbnRlbnRbNF07XG4gICAgICAgIHZhciB0cyA9IChkYXRhLmNvbnRlbnRbMF0gLSBOVFBfRVBPQ0hfREVMVEEpKjEwMDA7XG4gICAgICAgIHZhciBjaGFubmVsID0gZGF0YS5jb250ZW50WzJdO1xuICAgICAgICB2YXIgc2Vzc2lvblRzID0gZGF0YS5jb250ZW50WzFdO1xuXG4gICAgICAgIHRoaXMuYWRkTm90ZShub3RlLCB0cywgc2Vzc2lvblRzLCB2ZWxvY2l0eSwgY2hhbm5lbCwgMCk7XG4gICAgfTtcblxuICAgIHRoaXMuYWRkTm90ZSA9IGZ1bmN0aW9uKG5vdGUsIHN0YXJ0VGltZSwgc2Vzc2lvblRzLCB2ZWxvY2l0eSwgY2hhbm5lbCwgZHVyYXRpb24pIHtcblxuICAgICAgICB2YXIgdHMgPSBzdGFydFRpbWU7XG4gICAgICAgIGlmKHRoaXMub2Zmc2V0TXVzaWMpIHtcbiAgICAgICAgICAgIHRzID0gdGhpcy5zdGFydFRzICsgc2Vzc2lvblRzO1xuICAgICAgICB9XG5cbiAgICAgICAgdmFyIG5vdGVEdXJhdGlvbiA9IGR1cmF0aW9uO1xuICAgICAgICB2YXIgbm90ZVZlbG9jaXR5ID0gdmVsb2NpdHk7XG4gICAgICAgIHZhciBncmFwaGljcztcbiAgICAgICAgaWYoIWR1cmF0aW9uKSB7XG4gICAgICAgICAgICBpZih0eXBlb2YgdGhpcy5ub3RlRGljdFtjaGFubmVsXT09PSd1bmRlZmluZWQnKXtcbiAgICAgICAgICAgICAgICB0aGlzLm5vdGVEaWN0W2NoYW5uZWxdID0ge307XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBpZih2ZWxvY2l0eT09PTApIHtcbiAgICAgICAgICAgICAgICBpZih0eXBlb2YgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXSAhPT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgICAgICAgICAgICAgICAgdmFyIG5vdGVEZWYgPSB0aGlzLm5vdGVEaWN0W2NoYW5uZWxdW25vdGVdO1xuICAgICAgICAgICAgICAgICAgICBkZWxldGUgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXTtcbiAgICAgICAgICAgICAgICAgICAgbm90ZUR1cmF0aW9uID0gc2Vzc2lvblRzIC0gbm90ZURlZi5zZXNzaW9uVHM7XG4gICAgICAgICAgICAgICAgICAgIGdyYXBoaWNzID0gbm90ZURlZi5ncmFwaGljcztcbiAgICAgICAgICAgICAgICAgICAgbm90ZVZlbG9jaXR5ID0gbm90ZURlZi52ZWxvY2l0eTtcbiAgICAgICAgICAgICAgICAgICAgdHMgPSBub3RlRGVmLnRzO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgIG5vdGVEdXJhdGlvbiA9IE1hdGguYWJzKERhdGUubm93KCkgLSB0cyk7XG4gICAgICAgICAgICAgICAgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXSA9IHsgdHM6IHRzLCB2ZWxvY2l0eTogdmVsb2NpdHksIHNlc3Npb25Uczogc2Vzc2lvblRzfTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG5cbiAgICAgICAgaWYoIXRoaXMub2Zmc2V0TXVzaWMgfHwgdmVsb2NpdHk9PT0wKSB7XG5cbiAgICAgICAgICAgIHZhciB3aWR0aCA9IG5vdGVEdXJhdGlvbiAqIHRoaXMucGl4ZWxzUGVyU2Vjb25kIC8gMTAwMDtcbiAgICAgICAgICAgIGlmKCFncmFwaGljcykge1xuICAgICAgICAgICAgICAgIHZhciB4ID0gKHRzLXRoaXMuc3RhcnRUcykgKiB0aGlzLnBpeGVsc1BlclNlY29uZCAvIDEwMDA7XG4gICAgICAgICAgICAgICAgaWYoKHgrd2lkdGgpIDwgIChNYXRoLmFicyh0aGlzLmNvbnRhaW5lci54KSAtIHRoaXMud2lkdGgpKSB7XG4gICAgICAgICAgICAgICAgICAgIC8vIG5vdCB2aXNpYmxlLiBkbyBub3RoaW5nXG4gICAgICAgICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICBcdGlmICh0aGlzLmR5bmFtaWNSYW5nZSAmJiAodGhpcy5yYW5nZS5ib3R0b20gPiBub3RlIHx8IG5vdGUgPiB0aGlzLnJhbmdlLnRvcCkpe1xuICAgICAgICAgICAgXHRcdHZhciBuZXdTY2FsZSA9IHt9O1xuICAgICAgICAgICAgXHRcdG5ld1NjYWxlWydib3R0b20nXSA9IE1hdGgubWluKG5vdGUsIHRoaXMucmFuZ2UuYm90dG9tKTtcbiAgICAgICAgICAgIFx0XHRuZXdTY2FsZVsndG9wJ10gPSBNYXRoLm1heChub3RlLCB0aGlzLnJhbmdlLnRvcCk7XG4gICAgICAgICAgICBcdFx0dGhpcy5yZXNjYWxlU2NlbmUobmV3U2NhbGUpO1xuICAgICAgICAgICAgXHR9XG4gICAgICAgICAgICBcdHZhciB5ID0gTWF0aC5mbG9vcigoKHRoaXMucmFuZ2UudG9wLXRoaXMucmFuZ2UuYm90dG9tKS0obm90ZS10aGlzLnJhbmdlLmJvdHRvbSkrMC41KSAqICh0aGlzLm5vdGVIZWlnaHQpIC0gKHRoaXMubm90ZUhlaWdodC8yKSk7XG4gICAgICAgICAgICAgICAgdmFyIGNvbG9yID0gdGhpcy5nZXRDb2xvcihjaGFubmVsKTtcbiAgICAgICAgICAgICAgICBcbiAgICAgICAgICAgICAgICB2YXIgYWxwaGEgPSAobm90ZVZlbG9jaXR5IC8gMTI4KTtcblxuICAgICAgICAgICAgICAgIGdyYXBoaWNzID0gdGhpcy5nZXROb3RlUmVjdChub3RlLCB4LCB5LCBjb2xvciwgYWxwaGEsIHdpZHRoLCB0aGlzLm5vdGVIZWlnaHQpO1xuICAgICAgICAgICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGdyYXBoaWNzKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgIGdyYXBoaWNzLndpZHRoID0gd2lkdGg7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmKCFkdXJhdGlvbiAmJiB2ZWxvY2l0eSkge1xuICAgICAgICAgICAgICAgIHRoaXMubm90ZURpY3RbY2hhbm5lbF1bbm90ZV0uZ3JhcGhpY3MgPSBncmFwaGljcztcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgIH07XG4gICAgXG4gICAgLy9yZXNjYWxlIHNjZW5lIGluIGNhc2UgYSBub3RlIG91dCBvZiByYW5nZSBpcyBhZGRlZCBvciBhIG5vdGUgb3V0IG9mIHRoZSByYW5nZSBpcyByZW1vdmVkXG4gICAgdGhpcy5yZXNjYWxlU2NlbmUgPSBmdW5jdGlvbihuZXdTY2FsZSl7XG4gICAgXHR2YXIgX3RoaXMgPSB0aGlzO1xuICAgIFx0dmFyIGNoaWxkcmVuVG9VcGRhdGUgPSBbXTtcbiAgICBcdHZhciB0b3AgPSB0aGlzLmluaXRSYW5nZS50b3A7XG4gICAgXHR2YXIgYm90dG9tID0gdGhpcy5pbml0UmFuZ2UuYm90dG9tO1xuICAgIFx0XyhfdGhpcy5jb250YWluZXIuY2hpbGRyZW4pLmZvckVhY2goZnVuY3Rpb24oY2hpbGQpIHtcbiAgICBcdFx0aWYgKHR5cGVvZihjaGlsZCkgIT09ICd1bmRlZmluZWQnICYmIGNoaWxkLm5vdGUgJiYgIWlzSGlkZGVuKGNoaWxkKSl7XG4gICAgXHRcdFx0dG9wID0gTWF0aC5tYXgoY2hpbGQubm90ZSwgdG9wKTtcbiAgICBcdFx0XHRib3R0b20gPSBNYXRoLm1pbihjaGlsZC5ub3RlLCBib3R0b20pO1xuICAgIFx0XHRcdHJldHVybiBjaGlsZHJlblRvVXBkYXRlLnB1c2goY2hpbGQpO1xuICAgIFx0XHR9XG4gICAgICAgIH0pO1xuICAgIFx0aWYgKG5ld1NjYWxlKXtcbiAgICBcdFx0dGhpcy5yYW5nZSA9IG5ld1NjYWxlO1xuICAgIFx0fWVsc2V7XG4gICAgXHRcdHRoaXMucmFuZ2UudG9wID0gdG9wOyBcbiAgICBcdFx0dGhpcy5yYW5nZS5ib3R0b20gPSBib3R0b207IFxuICAgIFx0fVxuICAgIFx0dGhpcy5ub3RlSGVpZ2h0ID0gdGhpcy5oZWlnaHQgLyAodGhpcy5yYW5nZS50b3AgLSB0aGlzLnJhbmdlLmJvdHRvbSArIDEpO1xuICAgIFx0Y2hpbGRyZW5Ub1VwZGF0ZS5mb3JFYWNoKGZ1bmN0aW9uKGNoaWxkKSB7XG5cdFx0XHRjaGlsZC55ID0gTWF0aC5mbG9vcigoKF90aGlzLnJhbmdlLnRvcC1fdGhpcy5yYW5nZS5ib3R0b20pLShjaGlsZC5ub3RlLV90aGlzLnJhbmdlLmJvdHRvbSkrMC41KSAqIChfdGhpcy5ub3RlSGVpZ2h0KSAtIChfdGhpcy5ub3RlSGVpZ2h0LzIpKTtcblx0XHRcdGNoaWxkLmhlaWdodCA9IF90aGlzLm5vdGVIZWlnaHQ7XG4gICAgICAgIH0pO1xuICAgIH07XG5cbiAgICB0aGlzLmFkZExpbmUgPSBmdW5jdGlvbih0cyl7XG5cbiAgICAgICAgaWYodHlwZW9mKHRzKSA9PT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgICAgICAgIHRzID0gbmV3IERhdGUoKTtcbiAgICAgICAgfVxuICAgICAgICB2YXIgeCA9IC10aGlzLmNvbnRhaW5lci54O1xuICAgICAgICB2YXIgeSA9IHRoaXMubGluZXNEb3duID8gdGhpcy5oZWlnaHQgLSAyMCA6IDA7XG5cbiAgICAgICAgdmFyIGdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKVxuICAgICAgICAgICAgLmJlZ2luRmlsbCgweEZGRkYwMClcbiAgICAgICAgICAgIC5saW5lU3R5bGUoMSwgdGhpcy5saW5lQ29sb3IpXG4gICAgICAgICAgICAubW92ZVRvKDAsIDApXG4gICAgICAgICAgICAubGluZVRvKDAsIDIwKVxuICAgICAgICAgICAgLmVuZEZpbGwoKTtcbiAgICAgICAgZ3JhcGhpY3MueCA9IHg7XG4gICAgICAgIGdyYXBoaWNzLnkgPSB5O1xuICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChncmFwaGljcyk7XG4gICAgICAgIC8vIEFkZCB0ZXh0XG4gICAgICAgIC8vdmFyIHRvdGFsU2VjID0gbGluZU5iICogdGhpcy5saW5lSW50ZXJ2YWwgLyAxMDAwO1xuICAgICAgICB2YXIgaG91cnMgPSB0cy5nZXRIb3VycygpO1xuICAgICAgICB2YXIgbWludXRlcyA9dHMuZ2V0TWludXRlcygpO1xuICAgICAgICB2YXIgc2Vjb25kcyA9IHRzLmdldFNlY29uZHMoKTtcbiAgICAgICAgdmFyIHRpbWVTdHIgPSAoaG91cnMgPCAxMCA/ICcwJyArIGhvdXJzIDogaG91cnMpICsgJzonICsgKG1pbnV0ZXMgPCAxMCA/ICcwJyArIG1pbnV0ZXMgOiBtaW51dGVzKSArICc6JyArIChzZWNvbmRzICA8IDEwID8gJzAnICsgc2Vjb25kcyA6IHNlY29uZHMpO1xuXG4gICAgICAgIHZhciBmb250T2JqID0geyBmb250OiAnMTBwdCBBcmlhbCcsIGZpbGw6ICcjNDQ0NDQ0JyB9O1xuICAgICAgICB2YXIgdCA9IG5ldyBQSVhJLlRleHQodGltZVN0ciwgZm9udE9iaik7XG4gICAgICAgIGlmKGlzSG9yaXpvbnRhbCkge1xuICAgICAgICAgICAgdC54ID0geCArIDI7XG4gICAgICAgICAgICB0LnkgPSB0aGlzLmxpbmVzRG93biA/IHRoaXMuaGVpZ2h0IC0gMTUgOiAyO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgdC5yb3RhdGlvbiA9IC1NYXRoLlBJLzI7XG4gICAgICAgICAgICB0LnggPSB4IDtcbiAgICAgICAgICAgIHQueSA9IHRoaXMubGluZXNEb3duID8gdGhpcy5oZWlnaHQgLSAyIDogdC53aWR0aCArIDI7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodCk7XG4gICAgfTtcblxuICAgIHRoaXMubW92ZVRvID0gZnVuY3Rpb24oZGlmZlRpbWUpe1xuICAgICAgICB2YXIgb2xkWCA9IHRoaXMuY29udGFpbmVyLng7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLnggPSBNYXRoLmZsb29yKGRpZmZUaW1lKnRoaXMucGl4ZWxzUGVyU2Vjb25kKTtcbiAgICAgICAgdmFyIGRlbHRhWCA9IE1hdGguYWJzKG9sZFgtdGhpcy5jb250YWluZXIueCk7XG4gICAgICAgIF8uZm9yT3duKHRoaXMubm90ZURpY3QsIGZ1bmN0aW9uKGNoYW5uZWxEaWN0KSB7XG4gICAgICAgICAgICBfLmZvck93bihjaGFubmVsRGljdCwgZnVuY3Rpb24obm90ZURlZikge1xuICAgICAgICAgICAgICAgIGlmKG5vdGVEZWYuZ3JhcGhpY3MpIHtcbiAgICAgICAgICAgICAgICAgICAgbm90ZURlZi5ncmFwaGljcy53aWR0aCA9IG5vdGVEZWYuZ3JhcGhpY3Mud2lkdGggKyBkZWx0YVg7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cbiAgICB0aGlzLm1vdmUgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgdmFyIGRpZmYgPSAodGhpcy5zdGFydFRzIC0gRGF0ZS5ub3coKSkvMTAwMDtcbiAgICAgICAgdGhpcy5tb3ZlVG8oZGlmZik7XG4gICAgfTtcblxuICAgIHRoaXMucmVtb3ZlUGFzc2VkT2JqZXRzID0gZnVuY3Rpb24oKXtcbiAgICAgICAgdmFyIGNoaWxkcmVuVG9SZW1vdmUgPSBbXTtcbiAgICAgICAgdmFyIHJlc2NhbGUgPSBmYWxzZTtcbiAgICAgICAgXyhfdGhpcy5jb250YWluZXIuY2hpbGRyZW4pLmZvckVhY2goZnVuY3Rpb24oY2hpbGQpIHtcbiAgICAgICAgICAgIHJldHVybiB0eXBlb2YoY2hpbGQpID09PSAndW5kZWZpbmVkJyB8fFxuICAgICAgICAgICAgICAgIChpc0hpZGRlbihjaGlsZCkgJiYgY2hpbGRyZW5Ub1JlbW92ZS5wdXNoKGNoaWxkKSk7XG4gICAgICAgIH0pO1xuICAgICAgICBjaGlsZHJlblRvUmVtb3ZlLmZvckVhY2goZnVuY3Rpb24oY2hpbGQpIHtcbiAgICAgICAgXHRpZiAoX3RoaXMuZHluYW1pY1JhbmdlICYmIChfdGhpcy5yYW5nZS5ib3R0b20gPT09IGNoaWxkLm5vdGUgfHwgY2hpbGQubm90ZSA9PT0gX3RoaXMucmFuZ2UudG9wKSl7XG4gICAgICAgICAgICAgICAgcmVzY2FsZSA9IHRydWU7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBfdGhpcy5jb250YWluZXIucmVtb3ZlQ2hpbGQoY2hpbGQpO1xuICAgICAgICB9KTtcbiAgICAgICAgLy9leHRlcm5hbGl6ZSB0aGlzIHRlc3QgdG8gYXZvaWQgcmVwZWF0ZWQgY2FsbCB0byB0aGUgZnVuY3Rpb24gcmVzY2FsZVNjZW5lIGluIHRoZSBwcmV2aW91cyBsb29wXG4gICAgICAgIGlmIChyZXNjYWxlKXtcbiAgICAgICAgICAgIF90aGlzLnJlc2NhbGVTY2VuZSgpO1xuICAgICAgICB9XG4gICAgfTtcblxuICAgIHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgaWYoIXN0YXJ0ZWQpIHtcbiAgICAgICAgICAgIHRoaXMuc3RhcnRUcyA9IERhdGUubm93KCk7XG4gICAgICAgICAgICB0aGlzLmFkZExpbmUoKTtcbiAgICAgICAgICAgIHN0YXJ0ZWQgPSB0cnVlO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMudmVydGljYWxMaW5lc0ludGVydmFsID0gc2V0SW50ZXJ2YWwoZnVuY3Rpb24oKSB7IF90aGlzLmFkZExpbmUoKTsgfSwgdGhpcy5saW5lSW50ZXJ2YWwpO1xuICAgICAgICB0aGlzLmNsZWFuSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbiAoKSB7IF90aGlzLnJlbW92ZVBhc3NlZE9iamV0cygpOyB9LCAxMDAwICogdGhpcy53aWR0aCAvIHRoaXMucGl4ZWxzUGVyU2Vjb25kICk7XG4gICAgfTtcblxuICAgIHRoaXMuc3RvcCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBjbGVhckludGVydmFsKHRoaXMudmVydGljYWxMaW5lc0ludGVydmFsKTtcbiAgICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLmNsZWFuSW50ZXJ2YWwpO1xuICAgIH07XG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSBQaWFub1JvbGw7XG4iLCIvKipcbiogc2NyaXB0cy9zdGFnZXZpZXcuanNcbipcbiogVGhpcyBpcyB0aGUgc3RhcnRpbmcgcG9pbnQgZm9yIHlvdXIgYXBwbGljYXRpb24uXG4qIFRha2UgYSBsb29rIGF0IGh0dHA6Ly9icm93c2VyaWZ5Lm9yZy8gZm9yIG1vcmUgaW5mb1xuKi9cblxuLyogZ2xvYmFsIGRvY3VtZW50OiBmYWxzZSAqL1xuXG4ndXNlIHN0cmljdCc7XG5cblxudmFyIFBJWEkgPSByZXF1aXJlKCdwaXhpJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xuXG52YXIgZGVmYXVsdENvbmZpZyA9IHtcbiAgICBleHRlcm5hbFJlZnJlc2g6IGZhbHNlLFxuICAgIGxvZ2dlcjogdW5kZWZpbmVkLFxuICAgIHNjZW5lV2lkdGg6IDEwMjQsXG4gICAgc2NlbmVIZWlnaHQ6IDc2OCxcbiAgICBmcmFtZXJhdGU6IDI1LFxuICAgIHNjZW5lQmdDb2xvcjogMHhGRkZGRkYsXG4gICAgY2FudmFzQ29udGFpbmVyOiAnY2FudmFzQ29udGFpbmVyJyxcbn07XG5cbmZ1bmN0aW9uIFN0YWdlVmlldyhvcHRpb25zKSB7XG5cbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0Q29uZmlnKS52YWx1ZSgpO1xuXG4gICAgdmFyIGV4dGVybmFsUmVmcmVzaCA9IG9wdHMuZXh0ZXJuYWxSZWZyZXNoO1xuXG4gICAgdGhpcy5sb2dnZXIgPSBvcHRzLmxvZ2dlcjtcbiAgICB0aGlzLmZyYW1lcmF0ZSA9IG9wdHMuZnJhbWVyYXRlO1xuICAgIHZhciBzY2VuZUJnQ29sb3IgPSBvcHRzLnNjZW5lQmdDb2xvcjtcbiAgICB2YXIgc2NlbmVXaWR0aCA9IG9wdHMuc2NlbmVXaWR0aDtcbiAgICB2YXIgc2NlbmVIZWlnaHQgPSBvcHRzLnNjZW5lSGVpZ2h0O1xuICAgIHZhciBjYW52YXNDb250YWluZXIgPSBvcHRzLmNhbnZhc0NvbnRhaW5lcjtcbiAgICB2YXIgdGltZUNvbnRhaW5lciA9IFtdO1xuICAgIHZhciBjb21wb25lbnRzID0gW107XG5cbiAgICAvL2NyZWF0ZSBhbiBuZXcgaW5zdGFuY2Ugb2YgYSBwaXhpIHN0YWdlXG4gICAgdGhpcy5zdGFnZSA9IG5ldyBQSVhJLlN0YWdlKHNjZW5lQmdDb2xvcik7XG4gICAgLy9jcmVhdGUgYSByZW5kZXJlciBpbnN0YW5jZS5cbiAgICB2YXIgcmVuZGVyZXIgPSBQSVhJLmF1dG9EZXRlY3RSZW5kZXJlcihzY2VuZVdpZHRoLCBzY2VuZUhlaWdodCk7XG5cbiAgICB0aGlzLmluaXQgPSBmdW5jdGlvbigpIHtcblxuICAgICAgICBpZih0eXBlb2YoY2FudmFzQ29udGFpbmVyKSA9PT0gJ3N0cmluZycpIHtcbiAgICAgICAgICAgIGNhbnZhc0NvbnRhaW5lciA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKGNhbnZhc0NvbnRhaW5lcik7XG4gICAgICAgIH1cbiAgICAgICAgaWYodHlwZW9mKHRpbWVDb250YWluZXIpID09PSAnc3RyaW5nJykge1xuICAgICAgICAgICAgdGltZUNvbnRhaW5lciA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKHRpbWVDb250YWluZXIpO1xuICAgICAgICB9XG5cbiAgICAgICAgY2FudmFzQ29udGFpbmVyLmFwcGVuZENoaWxkKHJlbmRlcmVyLnZpZXcpO1xuXG4gICAgICAgIGNvbXBvbmVudHMuZm9yRWFjaChmdW5jdGlvbihjKXtcbiAgICBcdFx0Yy5pbml0KCk7XG4gICAgXHR9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZWdpc3RlclRpbWVDb250YWluZXIgPSBmdW5jdGlvbihjb250YWluZXIpIHtcbiAgICBcdHRpbWVDb250YWluZXIucHVzaChjb250YWluZXIpO1xuICAgIH07XG5cbiAgICB0aGlzLnJlZ2lzdGVyQ29tcG9uZW50ID0gZnVuY3Rpb24oY29tcG9uZW50KSB7XG4gICAgXHRjb21wb25lbnRzLnB1c2goY29tcG9uZW50KTtcbiAgICBcdHRoaXMuc3RhZ2UuYWRkQ2hpbGQoY29tcG9uZW50LmNvbnRhaW5lcik7XG4gICAgfTtcblxuICAgIHRoaXMucmVmcmVzaCA9IGZ1bmN0aW9uKCkge1xuICAgIFx0Y29tcG9uZW50cy5mb3JFYWNoKGZ1bmN0aW9uKGMpe1xuICAgIFx0XHRjLnJlZnJlc2goKTtcbiAgICBcdH0pO1xuICAgICAgICByZW5kZXJlci5yZW5kZXIodGhpcy5zdGFnZSk7XG4gICAgfTtcblxuICAgIC8vIEluaXQgcGFnZSBhbmQgaW50ZXJ2YWxzXG4gICAgdmFyIHJlZnJlc2hJbnRlcnZhbDtcblxuICAgIHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcblxuICAgICAgICBpZighZXh0ZXJuYWxSZWZyZXNoKSB7XG4gICAgICAgICAgICByZWZyZXNoSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbigpIHtfdGhpcy5yZWZyZXNoKCk7fSwgMTAwMC90aGlzLmZyYW1lcmF0ZSk7XG4gICAgICAgIH1cblxuICAgICAgICBjb21wb25lbnRzLmZvckVhY2goZnVuY3Rpb24oYyl7XG4gICAgXHRcdGMuc3RhcnQoKTtcbiAgICBcdH0pO1xuICAgIH07XG5cbiAgICB0aGlzLnN0b3AgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgaWYoIWV4dGVybmFsUmVmcmVzaCkge1xuICAgICAgICAgICAgY2xlYXJJbnRlcnZhbChyZWZyZXNoSW50ZXJ2YWwpO1xuICAgICAgICB9XG4gICAgICAgIFxuICAgICAgICBjb21wb25lbnRzLmZvckVhY2goZnVuY3Rpb24oYyl7XG4gICAgXHRcdGMuc3RvcCgpO1xuICAgIFx0fSk7XG4gICAgfTtcblxuXG4gICAgdGhpcy5sb2cgPSBmdW5jdGlvbihtKSB7XG4gICAgICAgIGlmKHRoaXMubG9nZ2VyKSB7XG4gICAgICAgICAgICB0aGlzLmxvZ2dlci5sb2cobSk7XG4gICAgICAgIH1cbiAgICB9O1xuXG5cbiAgICByZXR1cm4gdGhpcztcbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgU3RhZ2VWaWV3OiBTdGFnZVZpZXdcbn07XG4iLCIvKipcbioganMvdXRpbHMuanNcbipcbiogYmFzaWMgdG9vbHNcbipcbiovXG4vKmpzaGludCBiaXR3aXNlOiBmYWxzZSovXG4vKmpzaGludCBjYW1lbGNhc2U6IGZhbHNlICovXG5cbid1c2Ugc3RyaWN0JztcblxudmFyIFBJWEkgPSByZXF1aXJlKCdwaXhpJyk7XG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xudmFyIG1vbWVudCA9IHJlcXVpcmUoJ21vbWVudCcpO1xuXG5mdW5jdGlvbiBmb3JtYXRUaW1lICh0cykge1xuICAgIHJldHVybiBtb21lbnQodHMpLmZvcm1hdCgnSEg6bW06c3MnKTtcbn1cblxuZnVuY3Rpb24gY29sb3JUb0hleChjKSB7XG4gICAgdmFyIG0gPSAvcmdiYT9cXChcXHMqKFxcZCspXFxzKixcXHMqKFxcZCspXFxzKixcXHMqKFxcZCspLy5leGVjKGMpO1xuICAgIHJldHVybiBtID8gJyMnICsgKDEgPDwgMjQgfCBtWzFdIDw8IDE2IHwgbVsyXSA8PCA4IHwgbVszXSkudG9TdHJpbmcoMTYpLnN1YnN0cigxKSA6IGM7XG59XG5cbmZ1bmN0aW9uIGdldEFubm90Q2F0ZWdvcmllcyh1cmxDYXRlZ29yaWVzLCBhbm5vdENhdGVnb3JpZXMpIHtcblxuICAgIHZhciBqc29uTG9hZGVyID0gbmV3IFBJWEkuSnNvbkxvYWRlcih1cmxDYXRlZ29yaWVzLCB0cnVlKTtcblxuICAgIGpzb25Mb2FkZXIub24oJ2xvYWRlZCcsIGZ1bmN0aW9uKHJlcykge1xuICAgICAgICB2YXIgZGF0YSA9IHJlcy50YXJnZXQuanNvbjtcblxuICAgICAgICB3aGlsZShhbm5vdENhdGVnb3JpZXMubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgYW5ub3RDYXRlZ29yaWVzLnBvcCgpO1xuICAgICAgICB9XG5cbiAgICAgICAgZGF0YS5zZXNzaW9ucy5mb3JFYWNoKGZ1bmN0aW9uKHNlc3Npb24pIHtcbiAgICAgICAgICAgIHZhciBhbm5vdENhdCA9IHtcbiAgICAgICAgICAgICAgICB0czogc2Vzc2lvbi5zdGFydF90cyA9PT0gbnVsbCA/IG5ldyBEYXRlKDApIDogRGF0ZS5wYXJzZShzZXNzaW9uLnN0YXJ0X3RzKSxcbiAgICAgICAgICAgICAgICBjb2xvcnM6IHt9XG4gICAgICAgICAgICB9O1xuICAgICAgICAgICAgdmFyIGNhdGVnb3JpZXNKc29uID0gc2Vzc2lvbi5jYXRlZ29yaWVzX2pzb247XG4gICAgICAgICAgICBhbm5vdENhdC5vcmRlciA9IGNhdGVnb3JpZXNKc29uLm9yZGVyO1xuICAgICAgICAgICAgaWYgKHR5cGVvZihhbm5vdENhdC5vcmRlclsnZGVmYXVsdCddKSA9PT0gJ3VuZGVmaW5lZCcpe1xuICAgICAgICAgICAgICAgIGFubm90Q2F0Lm9yZGVyLnB1c2goJ2RlZmF1bHQnKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHZhciBjYXRMaXN0ID0gXy5jbG9uZShjYXRlZ29yaWVzSnNvbi5jYXRlZ29yaWVzKTtcbiAgICAgICAgICAgIHdoaWxlKGNhdExpc3QubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgICAgIHZhciBjYXQgPSBjYXRMaXN0LnBvcCgpO1xuICAgICAgICAgICAgICAgIGlmKGNhdC5jb2RlKSB7XG4gICAgICAgICAgICAgICAgICAgIGFubm90Q2F0LmNvbG9yc1tjYXQuY29kZV0gPSBjb2xvclRvSGV4KGNhdC5jb2xvcik7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIGlmKGNhdC5zdWJjYXRlZ29yaWVzKSB7XG4gICAgICAgICAgICAgICAgICAgIGNhdExpc3QgPSBjYXRMaXN0LmNvbmNhdChjYXQuc3ViY2F0ZWdvcmllcyk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY2F0ZWdvcmllc0pzb24uY2F0ZWdvcmllcy5mb3JFYWNoKGZ1bmN0aW9uKGNhdCkge1xuICAgICAgICAgICAgICAgIGlmKGNhdC5jb2RlKSB7XG4gICAgICAgICAgICAgICAgICAgIGFubm90Q2F0LmNvbG9yc1tjYXQuY29kZV0gPSBjb2xvclRvSGV4KGNhdC5jb2xvcik7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBhbm5vdENhdC5kZWZhdWx0Q29sb3IgPSBjYXRlZ29yaWVzSnNvbi5kZWZhdWx0Q29sb3IgfHwgJyM1MzY5OTEnO1xuICAgICAgICAgICAgYW5ub3RDYXRlZ29yaWVzLnB1c2goYW5ub3RDYXQpO1xuICAgICAgICB9KTtcbiAgICB9KTtcblxuICAgIGpzb25Mb2FkZXIubG9hZCgpO1xuXG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIGZvcm1hdFRpbWU6IGZvcm1hdFRpbWUsXG4gICAgZ2V0QW5ub3RDYXRlZ29yaWVzOiBnZXRBbm5vdENhdGVnb3JpZXMsXG4gICAgY29sb3JUb0hleDogY29sb3JUb0hleFxufTtcbiIsIi8qKlxuKiBqcy93c3dyYXBwZXIuanNcbipcbiogc2ltcGxlIHdlYnNlcnZpY2Ugd3JhcHBlciB0byByZWdpc3RlciBjYWxsYmFja3Mgb24gb25tZXNzYWdlXG4qXG4qL1xuXG4vKiBnbG9iYWwgV2ViU29ja2V0OiBmYWxzZSAqL1xuXG4ndXNlIHN0cmljdCc7XG5cbmZ1bmN0aW9uIFdzV3JhcHBlcih3c3VybCwgbG9nZ2VyKSB7XG5cbiAgICB2YXIgdXJsID0gd3N1cmw7XG4gICAgdmFyIHNvY2sgPSBuZXcgV2ViU29ja2V0KHVybCk7XG4gICAgdmFyIGxvZ2dlck9iaiA9IGxvZ2dlcjtcblxuICAgIHZhciBsb2cgPSBmdW5jdGlvbihtc2cpIHtcbiAgICAgICAgaWYobG9nZ2VyT2JqKSB7XG4gICAgICAgICAgICBsb2dnZXJPYmoubG9nKG1zZyk7XG4gICAgICAgIH1cbiAgICB9O1xuXG4gICAgdmFyIGhhbmRsZXJzID0gW107XG5cbiAgICBzb2NrLm9ub3BlbiA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBsb2coJ0Nvbm5lY3RlZCB0byAnICsgdXJsKTtcbiAgICB9O1xuXG4gICAgc29jay5vbmNsb3NlID0gZnVuY3Rpb24oZSkge1xuICAgICAgICBsb2coJ0Nvbm5lY3Rpb24gY2xvc2VkICh3YXNDbGVhbiA9ICcgKyBlLndhc0NsZWFuICsgJywgY29kZSA9ICcgKyBlLmNvZGUgKyAnLCByZWFzb24gPSBcXCcnICsgZS5yZWFzb24gKyAnXFwnKScpO1xuICAgICAgICBzb2NrID0gbnVsbDtcbiAgICB9O1xuXG4gICAgc29jay5vbm1lc3NhZ2UgPSBmdW5jdGlvbihlKSB7XG4gICAgICAgIGxvZygncmVjZWl2ZWQgJyArIGUuZGF0YSk7XG4gICAgICAgIHZhciBkYXRhID0gSlNPTi5wYXJzZShlLmRhdGEpO1xuICAgICAgICBoYW5kbGVycy5mb3JFYWNoKGZ1bmN0aW9uKGhhbmRsZXIpIHtcbiAgICAgICAgICAgIGhhbmRsZXIoZGF0YSk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cbiAgICB0aGlzLm1lc3NhZ2UgPSBmdW5jdGlvbihoYW5kbGVyKSB7XG4gICAgICAgIGlmKGhhbmRsZXIpIHtcbiAgICAgICAgICAgIGhhbmRsZXJzLnB1c2goaGFuZGxlcik7XG4gICAgICAgIH1cbiAgICB9O1xuXG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIFdzV3JhcHBlcjogV3NXcmFwcGVyXG59O1xuIl19