From 434e308f246f94c212c7041d0de29b0599207062 Mon Sep 17 00:00:00 2001 From: nova Date: Sat, 7 Dec 2019 11:31:42 +0530 Subject: [PATCH] ts integrated and working preliminarily --- chatto/.gitignore | 3 +- chatto/config/application.properties | 6 +- chatto/src/main/resources/static/js/bundle.js | 1095 +++++++++-------- chatto/src/main/resources/static/js/login.js | 6 +- chatto/src/main/resources/static/js/scljs.js | 60 - chatto/src/main/resources/templates/chat.html | 20 +- .../resources/templates/fragments/head.html | 1 - 7 files changed, 601 insertions(+), 590 deletions(-) delete mode 100644 chatto/src/main/resources/static/js/scljs.js diff --git a/chatto/.gitignore b/chatto/.gitignore index ec9c31a..a95be67 100644 --- a/chatto/.gitignore +++ b/chatto/.gitignore @@ -30,4 +30,5 @@ build/ ### VS Code ### .vscode/ -src/main/javascript/node_modules \ No newline at end of file +src/main/javascript/node_modules +config/ diff --git a/chatto/config/application.properties b/chatto/config/application.properties index ed7ad66..a9159ad 100644 --- a/chatto/config/application.properties +++ b/chatto/config/application.properties @@ -1,8 +1,8 @@ ## Spring DATASOURCE (DataSourceAutoConfiguration & DataSourceProperties) -spring.datasource.url = jdbc:mysql://localhost:3306/chatto_db2?useSSL=false +spring.datasource.url = jdbc:mysql://localhost:3306/chatto_db?useSSL=false spring.datasource.username = chatto_user -spring.datasource.password = password -database-name = chatto_db2 +spring.datasource.password = test +database-name = chatto_db website-url = 192.168.1.13 test.bindAddress=192.168.1.106 diff --git a/chatto/src/main/resources/static/js/bundle.js b/chatto/src/main/resources/static/js/bundle.js index 9e79679..bbcead7 100644 --- a/chatto/src/main/resources/static/js/bundle.js +++ b/chatto/src/main/resources/static/js/bundle.js @@ -1,566 +1,627 @@ (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i 0) { -// current[0].className = current[0].className.replace(" active", ""); -// } - -// // Add the active class to the current/clicked button -// this.className += " active"; -// }); -// } +JsonAPI_1.JsonAPI.ACTIVE_USERS_GET + 'aef'; +const encryptionService = new SJCLEncryptionService_1.SJCLEncryptionService(); +let ct = encryptionService.encrypt("password", "data"); +console.log(encryptionService.decrypt("password", JSON.parse(ct))); +Handlebars.registerHelper('avatar', function () { + return '
'; +}); -getActiveUsers(authToken) - .then(data => { - // activeUsers = data; - sessionStorage.setItem('activeUsers', JSON.stringify(data)); - log.log(sessionStorage.getItem('activeUsers')); - }) +}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) -for (let i = 0; i < userBoxes.length; i++) { - userBoxes[i].addEventListener('click', userCallBack) -} +},{"./controller/ChatController":1,"./controller/UserController":2,"./model/ChatModel":5,"./model/UserModel":8,"./service/SJCLEncryptionService":9,"./singleton/JsonAPI":10,"./view/ChatView":13,"./view/UserView":14}],4:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); -function addUserCallBacks() { - for (let i = 0; i < userBoxes.length; i++) { - userBoxes[i].addEventListener('click', userCallBack) +},{}],5:[function(require,module,exports){ +(function (global){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const log = (typeof window !== "undefined" ? window['log'] : typeof global !== "undefined" ? global['log'] : null); +const ChatModelHelper_1 = require("./ChatModelHelper"); +class ChatModel { + constructor() { + /** + * @type {Observer[]} List of subscribers. In real life, the list of + * subscribers can be stored more comprehensively (categorized by event + * type, etc.). + */ + this._observers = []; + this.state = null; + this._messagesMap = new Map(); + } + /** + * The subscription management methods. + */ + attach(observer) { + console.log('Subject: Attached an observer.'); + this._observers.push(observer); + } + detach(observer) { + const observerIndex = this._observers.indexOf(observer); + this._observers.splice(observerIndex, 1); + console.log('Subject: Detached an observer.'); + } + setUserMessages(username, messages) { + this._messagesMap.set(username, messages); + } + /** + * Trigger an update in each subscriber. + */ + notify(userName) { + console.log('Subject: Notifying observers...'); + for (const observer of this._observers) { + observer.update(this._messagesMap.get(userName)); + } + } + someBusinessMethod(chatMessageList) { + this.state = chatMessageList; + this.helperMethod(); + console.log(`Subject: My state has just changed`); + console.log(chatMessageList); + this.notify("some user"); + } + async getmessages(userName, passphrase, lastMessageTime) { + const cVMs = await ChatModelHelper_1.ChatModelHelper.getMessages(userName, passphrase, lastMessageTime, this); + if (cVMs != null) { + log.info('Subject: My state has just changed'); + log.debug(cVMs); + this._messagesMap.set(userName, cVMs); + this.notify(userName); + } + else { + log.error('Messages were null'); + } + return cVMs; + } + helperMethod() { } + populateMessages() { } } +exports.ChatModel = ChatModel; -function userCallBack() { - let current = document.getElementsByClassName('user-box active'); - let passphrase = passphraseInput.value; - if (current.length > 0) { - if (passphrase == '') { - // alert('Please input passphrase') - alertify.error('Please enter a passphrase'); +}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) + +},{"./ChatModelHelper":6}],6:[function(require,module,exports){ +(function (global){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const log = (typeof window !== "undefined" ? window['log'] : typeof global !== "undefined" ? global['log'] : null); +const ChatMessageViewModel_1 = require("../viewmodel/ChatMessageViewModel"); +const JsonAPI_1 = require("../singleton/JsonAPI"); +const FetchErrorHandler_1 = require("./FetchErrorHandler"); +const SJCLEncryptionService_1 = require("../service/SJCLEncryptionService"); +class ChatModelHelper { + static async getMessages(userName, passphrase, lastMessageTime, chatModel) { + switch (lastMessageTime) { + case null: { + // this.getAllMessagesAjax(userName) + // .then((data: ChatMessageDTO[]) => { + // log.debug(`Subject: received all messages`); + // // let userNames = data.map(ChatMessageViewModel => ChatMessageViewModel.fromUser) + // // let sumt = data.map(chatMessageViewModel => { return this.encryptionService.decrypt(passphrase, chatMessageViewModel.messageCipher) }); + // return data.map(vm => this.toChatMessageVM(vm, passphrase)); + // // chatModel.setUserMessages(userName, chatMessageVMs); + // // chatModel.notify(); + // }) + // break; + const data = await this.getAllMessagesAjax(userName); + return data.map(vm => this.toChatMessageVM(vm, passphrase)); + } + default: { + // this.getNewMessagesAjax(userName, lastMessageTime) + // .then((data: ChatMessageDTO[]) => { + // log.debug(`Subject: received new messages`); + // return data.map(vm => this.toChatMessageVM(vm, passphrase)); + // // chatModel.setUserMessages(userName, chatMessageVMs); + // // this.state = data; + // // chatModel.notify(); + // }) + // break; + const data = await this.getNewMessagesAjax(userName, lastMessageTime); + return data.map(vm => this.toChatMessageVM(vm, passphrase)); + } + } + // return null; + } + static toChatMessageVM(chatMessageDTO, passphrase) { + const vm = new ChatMessageViewModel_1.ChatMessageViewModel(); + vm.fromUser = chatMessageDTO.fromUser; + vm.toUser = chatMessageDTO.toUser; + vm.messageTime = chatMessageDTO.messageTime; + vm.message = this.encryptionService.decrypt(passphrase, chatMessageDTO.messageCipher); + return vm; + } + static async getAllMessagesAjax(toUser) { + const headers = new Headers(); + if (JsonAPI_1.JsonAPI.authToken == null) { + log.error("authToken null"); return; } - current[0].className = current[0].className.replace(" active", ""); - + ; + headers.append('X-AUTH-TOKEN', JsonAPI_1.JsonAPI.authToken); + const response = await fetch(`${JsonAPI_1.JsonAPI.CHAT_MESSAGES_GET}/${toUser}`, { + method: 'GET', + headers: headers + }); + console.log(response.clone()); + if (FetchErrorHandler_1.fetchErrorHandler(response.clone())) { + return null; + } + const data = await response.json(); + return data; } - // Add the active class to the current/clicked button - else if (current.length == 0) { - let elem = document.getElementById('passphrase-initial'); - passphrase = elem.value; - if (passphrase == '') { - // alert('Please input passphrase') - alertify.error('Please enter a passphrase'); + static async getNewMessagesAjax(toUser, lastMessageTimeStamp) { + const headers = new Headers(); + if (JsonAPI_1.JsonAPI.authToken == null) { + log.error("authToken null"); return; } - document.getElementById('no-user-selected').hidden = true; - document.getElementById('chat-card').hidden = false; - elem.hidden = true; - } - // console.log(this.getElementsByClassName('to-user-span')); - let userName = this.getElementsByClassName('to-user-span')[0].innerText; - document.getElementById('user-name-span').innerText = userName; - populateMessages(userName, passphrase); - sessionStorage.setItem('selectedUser', userName); - this.className += " active"; -} - -function populateMessages(userName, passphrase) { - console.log('Selected user = ' + userName); - if (passphrase == '') { - alert('Please input passphrase') - return; - } - // console.log(userName); - if (sessionStorage.getItem(userName) == null) { - chatTextArea.textContent = ''; - chatAreaNew.innerHTML = ''; - getAllMessages(userName) - .then(json => { - if (json == null) return; - console.log(json); - let i = 0; - let messageLog = []; - let messageLogNew = []; - let lastMessageTimeStamp; - - if (json.length > 0) { - json.forEach(function(obj) { - // console.log(obj.toUser); - messageCipher = JSON.stringify(obj.messageCipher); - console.log(messageCipher); - // let message = sjcl.decrypt("password", messageCipher); - let message = md.render(sjcl.decrypt(passphrase, messageCipher)); - let utcDate = obj.messageTime; - lastMessageTimeStamp = utcDate; - let localDate = new Date(utcDate); - let messageLine = sprintf('%s %s: %s ', localDate, obj.fromUser, message); - - - // localDate.`` - // console.log('localDate = ' + localDate); - console.log(messageLine); - // chatTextArea.append(obj.fromUser + ": " + message + "\n"); - chatTextArea.append(messageLine + '\n'); - messageLog[i++] = messageLine; - chatTextArea.scrollTop = chatTextArea.scrollHeight; - // console.log('Message log = ' + messageLog); - - let context = { fromUser: obj.fromUser, message: message, time: localDate.toLocaleString() }; - let msgContainer; - if (obj.fromUser == username) { - msgContainer = msgContainerSendTemplate(context); - } else { - msgContainer = msgContainerTemplate(context); - } - - messageLogNew.push(JSON.stringify(context)); - $(chatAreaNew).append(DOMPurify.sanitize(msgContainer)); - - - }); - sessionStorage.setItem(userName, JSON.stringify(messageLog)); - sessionStorage.setItem(userName + username + 'new', JSON.stringify(messageLogNew)); - // console.log() - // sessionStorage.clear(); - console.log('Last message time = ' + lastMessageTimeStamp); - sessionStorage.setItem(userName + '-time', lastMessageTimeStamp); - - - } - }); - } else { - - - console.log("Stored messages = " + sessionStorage.getItem(userName)); - let storedMessages = JSON.parse(sessionStorage.getItem(userName)); - let storedMessagesNew = JSON.parse(sessionStorage.getItem(userName + username + 'new')); - let lastMessageTime = sessionStorage.getItem(userName + '-time'); - console.log("last message time stamp = " + lastMessageTime); - if (lastMessageTime != null) { - getNewMessages(userName, lastMessageTime) - .then(json => { - if (json == null) return; - console.log(json) - if (json.length > 0) { - json.forEach(function(obj) { - let messageCipher = JSON.stringify(obj.messageCipher); - let message = md.render(sjcl.decrypt(passphrase, messageCipher)); - // console.log(message); - // chatTextArea.append(message + "\n"); - let utcDate = obj.messageTime; - lastMessageTimeStamp = utcDate; - let localDate = new Date(utcDate); - let messageLine = sprintf('%s %s: %s', localDate, obj.fromUser, message); - - // localDate.`` - // console.log('localDate = ' + localDate); - console.log(messageLine); - // chatTextArea.append(obj.fromUser + ": " + message + "\n"); - chatTextArea.append(messageLine + '\n'); - chatTextArea.scrollTop = chatTextArea.scrollHeight; - storedMessages.push(messageLine); - - let context = { fromUser: obj.fromUser, message: message, time: localDate.toLocaleString() }; - let msgContainer; - if (obj.fromUser == username) { - msgContainer = msgContainerSendTemplate(context); - } else { - msgContainer = msgContainerTemplate(context); - } - - storedMessagesNew.push(JSON.stringify(context)); - $(chatAreaNew).append(DOMPurify.sanitize(msgContainer)); - - }) - sessionStorage.setItem(userName + '-time', lastMessageTimeStamp); - sessionStorage.setItem(userName, JSON.stringify(storedMessages)); - sessionStorage.setItem(userName + username + 'new', JSON.stringify(storedMessagesNew)); - console.log("this value stored" + sessionStorage.getItem(userName)) - console.log("last message time stamp = " + lastMessageTimeStamp); - console.log(sessionStorage.getItem(userName + '-time')); - - } - chatTextArea.textContent = ''; - chatAreaNew.innerHTML = ''; - console.log("Stored messages 2 = " + storedMessages); - storedMessages.forEach(function(messageLine) { - chatTextArea.append(messageLine + '\n'); - chatTextArea.scrollTop = chatTextArea.scrollHeight; - - // let context = {message: messageLine}; - // let msgContainer; - // if(obj.fromUser == username) - // { - // msgContainer = msgContainerSendTemplate(context); - // } - // else{ - // msgContainer = msgContainerTemplate(context); - // } - - // $(chatAreaNew).append(msgContainer); - }) - - storedMessagesNew.forEach(function(contextString) { - let context = JSON.parse(contextString); - let msgContainer; - if (context.fromUser == username) { - msgContainer = msgContainerSendTemplate(context); - } else { - msgContainer = msgContainerTemplate(context); - } - - $(chatAreaNew).append(DOMPurify.sanitize(msgContainer)); - scrollChatAreaAnimated(2400); - }) - - - }); - - + ; + headers.append('X-AUTH-TOKEN', JsonAPI_1.JsonAPI.authToken); + const response = await fetch(`${JsonAPI_1.JsonAPI.CHAT_MESSAGES_GET}/${toUser}/${lastMessageTimeStamp}`, { + method: 'GET', + headers: headers + }); + console.log(response.clone()); + if (FetchErrorHandler_1.fetchErrorHandler(response.clone())) { + return null; } - // chatTextArea.append(JSON.stringify(storedMessages)); - + const data = await response.json(); + return data; } - // sessionStorage.setItem('status', 'ready'); - // sessionStorage.setItem('userName', messageLog); - // console.log('Message log = ' + messageLog); - // } - - // let passphraseKey = userName + '-passphrase'; - // sessionStorage.setItem(passphraseKey, passphrase); - // console.log(sessionStorage.getItem(passphraseKey)); } +exports.ChatModelHelper = ChatModelHelper; +ChatModelHelper.encryptionService = new SJCLEncryptionService_1.SJCLEncryptionService(); -// var lastMessageTimeStamp; - -// console.log(authToken); -// 'Basic ' + btoa("hmm" + ":" + "hmm") - -Handlebars.registerHelper('avatar', function() { - return '
'; -}); - +}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) -// var user; -function getSelectedUser() { - for (var i = 0; i < toUserRadios.length; i++) { - if (toUserRadios[i].checked) { - let user = toUserRadios[i].value; - console.log('sending to user = ' + user); - isCheckedUser = true; - return user; - } +},{"../service/SJCLEncryptionService":9,"../singleton/JsonAPI":10,"../viewmodel/ChatMessageViewModel":16,"./FetchErrorHandler":7}],7:[function(require,module,exports){ +(function (global){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const log = (typeof window !== "undefined" ? window['log'] : typeof global !== "undefined" ? global['log'] : null); +const sprintf_js_1 = (typeof window !== "undefined" ? window['sprintf'] : typeof global !== "undefined" ? global['sprintf'] : null); +// import sprintf = require('sprintf-js').sprintf; +function fetchErrorHandler(response) { + // alertify.success('Current position : ' + alertify.get('notifier', 'position')); + if (!response.ok) { + return response.text().catch(err => { + // the status was not ok and there is no json body + // throw new Error(response.statusText); + // window.alert(sprintf('Some error occured. Http code is %s', response.status)); + // alertify.error(sprintf('Some error occured. Http code is %s', response.status)); + log.error(sprintf_js_1.sprintf('Some error occured. Http code is %s', response.status)); + log.error(); + return true; + }).then(json => { + // the status was not ok but there is a json body + // throw new Error(json.error.message); // example error message returned by a REST API + // window.alert(sprintf('Error: %s (Http code %s)', json, response.status)); + // alertify.error(sprintf('Some error occured. Http code is %s', response.status)); + log.error(sprintf_js_1.sprintf('Some error occured. Http code is %s', response.status)); + log.error(json); + return true; + }); } - } +exports.fetchErrorHandler = fetchErrorHandler; -function getSelectedUserNew() { - return sessionStorage.getItem('selectedUser'); -} - -document.getElementById('chatMessageForm').addEventListener('submit', function(e) { - let chatInput = document.getElementById('chatInput'); - e.preventDefault(); - let user = getSelectedUserNew(); +}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) - if (!this.checkValidity()) { - console.log("error"); - this.classList.add('was-validated'); - return; +},{}],8:[function(require,module,exports){ +(function (global){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const FetchErrorHandler_1 = require("./FetchErrorHandler"); +const JsonAPI_1 = require("../singleton/JsonAPI"); +const log = (typeof window !== "undefined" ? window['log'] : typeof global !== "undefined" ? global['log'] : null); +class UserModel { + // @ts-ignore: Cannot find name 'hostAddress'. + constructor() { + /** + * @type {Observer[]} List of subscribers. In real life, the list of + * subscribers can be stored more comprehensively (categorized by event + * type, etc.). + */ + this.observers = []; } - this.classList.add('was-validated'); - - if (user == null) { - // window.alert('please select a user'); - alertify.error('Please select a user'); - return; - } - let messageContent = chatInput.value; - let context = { fromUser: username, message: md.render(messageContent), time: new Date().toLocaleString() }; - let msgContainer = msgContainerSendTemplate(context); - $(chatAreaNew).append(DOMPurify.sanitize(msgContainer)); - scrollChatAreaAnimated(2400); - let messageCipher = sjcl.encrypt(passphraseInput.value, messageContent, { mode: "gcm", ts: 128, adata: "", iter: iterations }); - let messageCipherJson = JSON.parse(messageCipher); - let chatMessageDTO = { - "toUser": user, - "messageCipher": messageCipherJson - } - messageSend(JSON.stringify(chatMessageDTO)); -}) - -document.getElementById('user-search').addEventListener('submit', function(e) { - e.preventDefault(); - let contactsBox = document.getElementById('contacts-box'); - let temp = contactsBox.innerHTML; - // log.trace(temp); - let searchTerm = document.getElementById('user-search-term').value; - log.debug("search term value = " + searchTerm); - let list = JSON.parse(sessionStorage.getItem('activeUsers')); - log.debug("active users"); - log.debug(list); - let fuse = new Fuse(list, fuseOptions); - let searchResult = fuse.search(searchTerm); - populateContactsBox(contactsBox, searchResult); - addUserCallBacks(); - log.debug(searchResult); -}) -document.getElementById('user-search-term').addEventListener('input', function(e) { - e.preventDefault(); - if (this.value.length < 2) { - log.debug("inputted") - let cancelButton = document.getElementById('user-search-cancel'); - cancelButton.hidden = false; - } -}) -document.getElementById('user-search-cancel').addEventListener('click', function(e) { - e.preventDefault(); - let list = JSON.parse(sessionStorage.getItem('activeUsers')); - let contactsBox = document.getElementById('contacts-box'); - populateContactsBox(contactsBox,list); - addUserCallBacks(); - document.getElementById('user-search-term').value = ""; - this.hidden = true; -}) - -function populateContactsBox(contactsBox, list) -{ - let userContactBoxList = ""; - list.forEach(function(activeUser) { - log.debug(activeUser); - if (activeUser.online) { - userContactBoxList += userContactOnlineTemplate(activeUser); - } else { - userContactBoxList += userContactOfflineTemplate(activeUser); + /** + * The subscription management methods. + */ + attach(observer) { + console.log('Subject: Attached an observer.'); + this.observers.push(observer); + } + detach(observer) { + const observerIndex = this.observers.indexOf(observer); + this.observers.splice(observerIndex, 1); + console.log('Subject: Detached an observer.'); + } + /** + * Trigger an update in each subscriber. + */ + notify() { + console.log('Subject: Notifying observers...'); + for (const observer of this.observers) { + observer.update(this.state); + } + } + someBusinessMethod(activeuserList) { + this.state = activeuserList; + this.helperMethod(); + console.log(`Subject: My state has just changed`); + console.log(activeuserList); + this.notify(); + } + /** + * getActiveUsers + */ + getActiveUsers() { + if (JsonAPI_1.JsonAPI.authToken != null) { + this.getActiveUsersAjax(JsonAPI_1.JsonAPI.authToken, JsonAPI_1.JsonAPI.ACTIVE_USERS_GET) + .then(data => { + // // activeUsers = data; + // sessionStorage.setItem('activeUsers', JSON.stringify(data)); + // console.log(sessionStorage.getItem('activeUsers')); + console.log(`Subject: received ajax active users`); + this.state = data; + this.notify(); + }); + } + else { + log.error('Auth token is null'); + } + } + async getActiveUsersAjax(authToken2, URL) { + let headers = new Headers(); + // headers.append('Authorization', basicAuthToken); + headers.append('X-AUTH-TOKEN', authToken2); + let response = await fetch(JsonAPI_1.JsonAPI.ACTIVE_USERS_GET, { + method: 'GET', + headers: headers + }); + console.log(response.clone()); + if (FetchErrorHandler_1.fetchErrorHandler(response.clone())) { + return null; } - }) - contactsBox.innerHTML = userContactBoxList; + let data = await response.json(); + // return data; + return new Promise((resolve, reject) => { + if (data != null) + resolve(data); + else + reject('Response data null'); + }); + } + helperMethod() { } } +exports.UserModel = UserModel; +}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) - -// console.log('Credentials = ' + JSON.parse(sessionStorage.getItem('credentials'))); - - -function messageSend(chatMessageDTO) { - let headers = new Headers(); - // console.log("Token = " + btoa("hmm" + ":" + "hmm")) - - // headers.append('Accept','application/json') - headers.append('Content-Type', 'application/json'); - // headers.append('Authorization', basicAuthToken); - headers.append('X-AUTH-TOKEN', authToken); - fetch(postNewMessageUrl, { - method: 'POST', - headers: headers, - body: chatMessageDTO - }) - .then(response => { - console.log(response); - return response.clone(); - }) - .then(response => fetchHandler(response)); +},{"../singleton/JsonAPI":10,"./FetchErrorHandler":7}],9:[function(require,module,exports){ +(function (global){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const sjcl = (typeof window !== "undefined" ? window['sjcl'] : typeof global !== "undefined" ? global['sjcl'] : null); +class SJCLEncryptionService { + constructor() { + this.params = { mode: "gcm", ts: 128, adata: "", iter: 10000 }; + } + encrypt(passphrase, plainText) { + return sjcl.encrypt(passphrase, plainText, this.params); + } + decrypt(passphrase, cipher) { + // return sjcl.decrypt(passphrase, cipher as sjcl.SjclCipherEncrypted, undefined, undefined); + return sjcl.decrypt(passphrase, JSON.stringify(cipher), undefined, undefined); + } } +exports.SJCLEncryptionService = SJCLEncryptionService; +}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) -async function getAllMessages(toUser) { - let headers = new Headers(); - // headers.append('Accept','application/json') - // headers.append('Content-Type', 'application/json'); - // headers.append('Authorization', basicAuthToken); - headers.append('X-AUTH-TOKEN', authToken); - let response = await fetch(getAllMessagesUrl + toUser, { - method: 'GET', - headers: headers - }); - console.log(response); - if (fetchErrorHandler(response.clone())) { - return null; - } - // if (response.status == 440) { - // window.alert('Token has expired. Please login again'); - // return null; +},{}],10:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +var JsonAPI; +(function (JsonAPI) { + // @ts-ignore: Cannot find name 'hostAddress'. + JsonAPI.userName = localStorage.getItem('userName'); + JsonAPI.authToken = localStorage.getItem('authToken'); + JsonAPI.ACTIVE_USERS_GET = `/api/chat/get/active-users`; + JsonAPI.CHAT_MESSAGES_GET = `/api/chat/get/messages`; +})(JsonAPI = exports.JsonAPI || (exports.JsonAPI = {})); + +},{}],11:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +class TemplateFactory { + // static getTemplate(templateName: string): Handlebars.TemplateDelegate { + // switch (templateName) { + // case "user-contact-online-template": { + // // let source = document.getElementById("user-contact-online-template").innerHTML; + // // let msgContainerTemplate = Handlebars.compile(source); + // // return msgContainerTemplate; + // return this.createTemplate(templateName); + // } + // case "msg_container_send_template": { + // // let source = document.getElementById("msg_container_send_template").innerHTML; + // // let msgContainerTemplate = Handlebars.compile(source); + // // return msgContainerTemplate; + // return this.createTemplate(templateName); + // } + // case "msg_container_template": { + // // let source = document.getElementById("msg_container_send_template").innerHTML; + // // let msgContainerTemplate = Handlebars.compile(source); + // // return msgContainerTemplate; + // return this.createTemplate(templateName); + // } + // default: + // throw new Error('invalid template name'); + // } // } - let data = await response.json(); - return data; -} - -async function getNewMessages(toUser, lastMessageTimeStamp) { - let headers = new Headers(); - // headers.append('Authorization', basicAuthToken); - headers.append('X-AUTH-TOKEN', authToken); - let response = await fetch(`${getNewMessagesUrl}${toUser}/${lastMessageTimeStamp}`, { - method: 'GET', - headers: headers - }); - console.log(response.clone()); - if (fetchErrorHandler(response.clone())) { - return null; - } - let data = await response.json(); - return data; + static getTemplate(templateName) { + //@ts-ignore: Object is possibly 'null'. + let source = document.getElementById(templateName).innerHTML; + let msgContainerTemplate = Handlebars.compile(source); + return msgContainerTemplate; + } } +exports.TemplateFactory = TemplateFactory; +},{}],12:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); -async function getActiveUsers(authToken2) { - let headers = new Headers(); - // headers.append('Authorization', basicAuthToken); - headers.append('X-AUTH-TOKEN', authToken2); - let response = await fetch(getActiveUsersUrl, { - method: 'GET', - headers: headers - }); - console.log(response.clone()); - if (fetchErrorHandler(response.clone())) { - return null; - } - let data = await response.json(); - return data; +},{}],13:[function(require,module,exports){ +(function (global){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const TemplateFactory_1 = require("../template/TemplateFactory"); +const log = (typeof window !== "undefined" ? window['log'] : typeof global !== "undefined" ? global['log'] : null); +const DOMPurify = (typeof window !== "undefined" ? window['DOMPurify'] : typeof global !== "undefined" ? global['DOMPurify'] : null); +const markdownit = (typeof window !== "undefined" ? window['markdownit'] : typeof global !== "undefined" ? global['markdownit'] : null); +var md = new markdownit(); +class ChatView { + constructor(model, element) { + this._messageSendTemplate = TemplateFactory_1.TemplateFactory.getTemplate('msg_container_send_template'); + this._model = model; + this._element = element; + } + update(data) { + log.info('ChatView: updating view'); + let html = ""; + data.forEach((vm) => { + html += this._messageSendTemplate(vm); + }); + /** Very Important!!! + * Sanitizing HTML before displaying on webpage to prevent XSS attacks!! + */ + html = DOMPurify.sanitize(md.render(html)); + this._element.innerHTML = html; + log.debug(this._element.innerHTML); + } } +exports.ChatView = ChatView; -$(document).ready(function() { - $('#action_menu_btn').click(function() { - $('.action_menu').toggle(); - }); -}); - -function fetchHandler(response) { - if (response.ok) { - return response.json().then(json => { - // the status was ok and there is a json body - // return Promise.resolve({ json: json, response: response }); - alertify.success('Message sent succesfully' + sprintf(" (http code %d)", response.status)); - }).catch(err => { - // the status was ok but there is no json body - // return Promise.resolve({ response: response }); - alertify.success('Message sent succesfully' + sprintf(" (http code %d)", response.status)); - }); +}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) - } else { - return response.json().catch(err => { - // the status was not ok and there is no json body - // throw new Error(response.statusText); - alertify.error('Some error occured. Please try again.'); - }).then(json => { - // the status was not ok but there is a json body - // throw new Error(json.error.message); // example error message returned by a REST - let delay = alertify.get('notifier', 'delay'); - alertify.set('notifier', 'delay', 30); - let errorMessage = ""; - json.errors.forEach(function(data) { - errorMessage += sprintf("Field Name: %s \n Rejected value: %s \n Reason: %s \n", data.field_name, data.rejected_value, data.error_message); - }); - alertify.error(sprintf('There were errors in your message - %s', errorMessage)); - alertify.set('notifier', 'delay', delay); +},{"../template/TemplateFactory":11}],14:[function(require,module,exports){ +(function (global){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const TemplateFactory_1 = require("../template/TemplateFactory"); +const log = (typeof window !== "undefined" ? window['log'] : typeof global !== "undefined" ? global['log'] : null); +class UserView { + // private userBoxes: any[] = []; + constructor(model, chatModel, element) { + this._model = model; + this._chatModel = chatModel; + this._element = element; + } + update(data) { + let template = TemplateFactory_1.TemplateFactory.getTemplate('user-contact-online-template'); + let html = ""; + data.forEach((element) => { + html += template(element); }); + this._element.innerHTML = html; + this.addUserCallBacks(); + console.log(this._element.innerHTML); + } + helper() { + } + addUserCallBacks() { + let userBoxes = document.getElementsByClassName('user-box'); + for (let i = 0; i < userBoxes.length; i++) { + let userBox = userBoxes[i]; + userBoxes[i].addEventListener('click', this.userCallBack.bind(this, userBox)); + } + } + userCallBack(el) { + let current = document.getElementsByClassName('user-box active'); + let passphrase = ''; + if (current.length > 0) { + let passphraseInput = document.getElementById('passphrase'); + if (passphraseInput == null) { + log.error('passphraseInput element reference is null'); + return; + } + passphrase = passphraseInput.value; + if (passphrase == '' || passphrase == null) { + // alert('Please input passphrase') + // alertify.error('Please enter a passphrase'); + log.error('passphrase is empty or null'); + return; + } + current[0].className = current[0].className.replace(" active", ""); + } + // Add the active class to the current/clicked button + else if (current.length == 0) { + let elem = document.getElementById('passphrase-initial'); + if (elem == null) { + log.error('passphraseInput element reference is null'); + return; + } + passphrase = elem.value; + if (passphrase == '' || passphrase == null) { + // // alert('Please input passphrase') + // // alertify.error('Please enter a passphrase'); + log.error('passphrase is empty or null'); + return; + } + // @ts-ignore: Object is possibly 'null'. + document.getElementById('no-user-selected').hidden = true; + // @ts-ignore: Object is possibly 'null'. + document.getElementById('chat-card').hidden = false; + // @ts-ignore: Object is possibly 'null'. + elem.hidden = true; + } + // console.log(this.getElementsByClassName('to-user-span')); + let elem = el.getElementsByClassName('to-user-span')[0]; + let userName = elem.innerText; + // @ts-ignore: Object is possibly 'null'. + document.getElementById('user-name-span').innerText = userName; + this._chatModel.getmessages(userName, passphrase, null); + // populateMessages(userName, passphrase); + sessionStorage.setItem('selectedUser', userName); + el.className += " active"; } } +exports.UserView = UserView; +}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) -function fetchErrorHandler(response) { - // alertify.success('Current position : ' + alertify.get('notifier', 'position')); - if (!response.ok) { - return response.text().catch(err => { - // the status was not ok and there is no json body - // throw new Error(response.statusText); - // window.alert(sprintf('Some error occured. Http code is %s', response.status)); - alertify.error(sprintf('Some error occured. Http code is %s', response.status)); - return true; - }).then(json => { - // the status was not ok but there is a json body - // throw new Error(json.error.message); // example error message returned by a REST API - // window.alert(sprintf('Error: %s (Http code %s)', json, response.status)); - alertify.error(sprintf('Some error occured. Http code is %s', response.status)); - console.log(json); - return true; - }); - } +},{"../template/TemplateFactory":11}],15:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +class ActiveUserViewModel { } +exports.ActiveUserViewModel = ActiveUserViewModel; -function scrollChatAreaAnimated(delay) { - $(chatAreaNew).stop().animate({ - scrollTop: $(chatAreaNew)[0].scrollHeight - }, delay); +},{}],16:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +class ChatMessageViewModel { } -}).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) -},{"./test.js":2}],2:[function(require,module,exports){ -log.info("test success"); -},{}]},{},[1]); +exports.ChatMessageViewModel = ChatMessageViewModel; + +},{}]},{},[3]) +//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Vzci9sb2NhbC9saWIvbm9kZV9tb2R1bGVzL3dhdGNoaWZ5L25vZGVfbW9kdWxlcy9icm93c2VyLXBhY2svX3ByZWx1ZGUuanMiLCJ0cy9zcmMvY29udHJvbGxlci9DaGF0Q29udHJvbGxlci50cyIsInRzL3NyYy9jb250cm9sbGVyL1VzZXJDb250cm9sbGVyLnRzIiwidHMvc3JjL21haW4udHMiLCJ0cy9zcmMvbW9kZWwvQ2hhdE1vZGVsLnRzIiwidHMvc3JjL21vZGVsL0NoYXRNb2RlbEhlbHBlci50cyIsInRzL3NyYy9tb2RlbC9GZXRjaEVycm9ySGFuZGxlci50cyIsInRzL3NyYy9tb2RlbC9Vc2VyTW9kZWwudHMiLCJ0cy9zcmMvc2VydmljZS9TSkNMRW5jcnlwdGlvblNlcnZpY2UudHMiLCJ0cy9zcmMvc2luZ2xldG9uL0pzb25BUEkudHMiLCJ0cy9zcmMvdGVtcGxhdGUvVGVtcGxhdGVGYWN0b3J5LnRzIiwidHMvc3JjL3ZpZXcvQ2hhdFZpZXcudHMiLCJ0cy9zcmMvdmlldy9Vc2VyVmlldy50cyIsInRzL3NyYy92aWV3bW9kZWwvQWN0aXZlVXNlclZpZXdNb2RlbC50cyIsInRzL3NyYy92aWV3bW9kZWwvQ2hhdE1lc3NhZ2VWaWV3TW9kZWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztBQ0NBLGtDQUErQjtBQUMvQiw4QkFBMkI7QUFDM0IsZ0NBQTZCO0FBQzdCLDRCQUF5QjtBQUd6Qiw0RUFBeUU7QUFJekUsTUFBYSxjQUFjO0lBS3ZCLFlBQVksS0FBZ0IsRUFBRSxJQUFjO1FBQ3hDLElBQUksQ0FBQyxNQUFNLEdBQUcsS0FBSyxDQUFDO1FBQ3BCLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDO0lBQ3RCLENBQUM7SUFHRDs7T0FFRztJQUNJLFlBQVksQ0FBQyxFQUEwQjtRQUMxQyxJQUFJLENBQUMsTUFBTSxDQUFDLGtCQUFrQixDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7SUFFTSxJQUFJO1FBQ1AsTUFBTSxxQkFBcUIsR0FBMkIsRUFBRSxDQUFDO1FBQ3pELElBQUksd0JBQXdCLEdBQUcsSUFBSSwyQ0FBb0IsRUFBRSxDQUFDO1FBQzFELHdCQUF3QixDQUFDLFFBQVEsR0FBRyxPQUFPLENBQUM7UUFDNUMsd0JBQXdCLENBQUMsTUFBTSxHQUFHLE9BQU8sQ0FBQztRQUMxQyx3QkFBd0IsQ0FBQyxPQUFPLEdBQUcsRUFBRSxDQUFDO1FBQ3RDLHdCQUF3QixDQUFDLFdBQVcsR0FBRyxJQUFJLElBQUksRUFBRSxDQUFDO1FBQ2xELHFCQUFxQixDQUFDLElBQUksQ0FBQyx3QkFBd0IsQ0FBQyxDQUFDO0lBQ3pELENBQUM7Q0FHSjtBQTdCRCx3Q0E2QkM7Ozs7O0FDdkNELGtDQUErQjtBQUMvQiw4QkFBMkI7QUFDM0IsZ0NBQTZCO0FBQzdCLDRCQUF5QjtBQUd6QiwwRUFBdUU7QUFJdkUsTUFBYSxjQUFjO0lBS3ZCLFlBQVksS0FBZ0IsRUFBRSxJQUFjO1FBQ3hDLElBQUksQ0FBQyxNQUFNLEdBQUcsS0FBSyxDQUFDO1FBQ3BCLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDO0lBQ3RCLENBQUM7SUFHRDs7T0FFRztJQUNJLFlBQVksQ0FBQyxFQUF5QjtRQUN6QyxJQUFJLENBQUMsTUFBTSxDQUFDLGtCQUFrQixDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7SUFFTSxJQUFJO1FBQ1AsTUFBTSxlQUFlLEdBQTBCLEVBQUUsQ0FBQztRQUNsRCxJQUFJLHVCQUF1QixHQUFHLElBQUkseUNBQW1CLEVBQUUsQ0FBQztRQUN4RCx1QkFBdUIsQ0FBQyxRQUFRLEdBQUcsV0FBVyxDQUFDO1FBQy9DLHVCQUF1QixDQUFDLFVBQVUsR0FBRyxXQUFXLENBQUM7UUFDakQsdUJBQXVCLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQztRQUN0QyxlQUFlLENBQUMsSUFBSSxDQUFDLHVCQUF1QixDQUFDLENBQUM7UUFDOUMsdUJBQXVCLEdBQUcsSUFBSSx5Q0FBbUIsRUFBRSxDQUFDO1FBQ3BELHVCQUF1QixDQUFDLFVBQVUsR0FBRyxXQUFXLENBQUM7UUFDakQsdUJBQXVCLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQztRQUN0Qyx1QkFBdUIsQ0FBQyxRQUFRLEdBQUcsYUFBYSxDQUFDO1FBQ2pELGVBQWUsQ0FBQyxJQUFJLENBQUMsdUJBQXVCLENBQUMsQ0FBQztRQUM5QyxJQUFJLENBQUMsWUFBWSxDQUFDLGVBQWUsQ0FBQyxDQUFDO0lBQ3ZDLENBQUM7SUFFTSxjQUFjO1FBQ2pCLElBQUksQ0FBQyxNQUFNLENBQUMsY0FBYyxFQUFFLENBQUM7SUFDakMsQ0FBQztDQUdKO0FBdENELHdDQXNDQzs7Ozs7O0FDaERELGlEQUE2QztBQUc3Qyw4Q0FBMkM7QUFDM0MsZ0VBQTZEO0FBSTdELHlDQUF5QztBQUN6QywwQ0FBMkM7QUFDM0MsaURBQThDO0FBQzlDLDhDQUEyQztBQUMzQyxnRUFBNkQ7QUFDN0QsaURBQThDO0FBQzlDLG1DQUFtQztBQUNuQyxnQ0FBZ0M7QUFHaEMsMkVBQXdFO0FBQ3hFLDJDQUEyQztBQUMzQyxJQUFJLEVBQUUsR0FBRyxJQUFJLFVBQVUsRUFBRSxDQUFDO0FBRzFCLE1BQU0sT0FBTyxHQUFHLFFBQVEsQ0FBQyxjQUFjLENBQUMsY0FBYyxDQUFDLENBQUM7QUFFeEQsR0FBRyxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQTtBQUVyQixNQUFNLFNBQVMsR0FBRyxJQUFJLHFCQUFTLEVBQUUsQ0FBQztBQUVsQyxNQUFNLFNBQVMsR0FBRyxJQUFJLHFCQUFTLEVBQUUsQ0FBQztBQUNsQyxzREFBc0Q7QUFDdEQsK0pBQStKO0FBQy9KLE1BQU0sUUFBUSxHQUFHLElBQUksbUJBQVEsQ0FBQyxTQUFTLEVBQUUsU0FBUyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0FBRTdELHdCQUF3QjtBQUV4QixTQUFTLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0FBQzNCLGlCQUFpQjtBQUdqQixNQUFNLGNBQWMsR0FBRyxJQUFJLCtCQUFjLENBQUMsU0FBUyxFQUFFLFFBQVEsQ0FBQyxDQUFDO0FBQy9ELHlCQUF5QjtBQUl6QixpREFBaUQ7QUFDakQsR0FBRyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztBQUVsQixNQUFNLFFBQVEsR0FBRyxRQUFRLENBQUMsY0FBYyxDQUFDLGVBQWUsQ0FBQyxDQUFDO0FBQzFELCtKQUErSjtBQUMvSixNQUFNLFFBQVEsR0FBRyxJQUFJLG1CQUFRLENBQUMsU0FBUyxFQUFFLFFBQVEsQ0FBQyxDQUFDO0FBQ25ELFNBQVMsQ0FBQyxNQUFNLENBQUMsUUFBUSxDQUFDLENBQUM7QUFDM0IsTUFBTSxjQUFjLEdBQUcsSUFBSSwrQkFBYyxDQUFDLFNBQVMsRUFBRSxRQUFRLENBQUMsQ0FBQztBQUcvRCxTQUFTLFFBQVEsQ0FBQyxFQUF1QjtJQUNyQyxnQkFBZ0I7SUFDaEIsa0JBQWtCO0FBQ3RCLENBQUM7QUFFRCxjQUFjLENBQUMsY0FBYyxFQUFFLENBQUM7QUFFaEMsR0FBRyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztBQUNqQixxQ0FBcUM7QUFFckMseUNBQXlDO0FBQ3pDLElBQUksTUFBTSxHQUFHLFFBQVEsQ0FBQyxjQUFjLENBQUMsd0JBQXdCLENBQUMsQ0FBQyxTQUFTLENBQUM7QUFFekUsSUFBSSxvQkFBb0IsR0FBRyxVQUFVLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0FBRXRELGlCQUFPLENBQUMsZ0JBQWdCLEdBQUcsS0FBSyxDQUFDO0FBRWpDLE1BQU0saUJBQWlCLEdBQXNCLElBQUksNkNBQXFCLEVBQUUsQ0FBQztBQUN6RSxJQUFJLEVBQUUsR0FBRyxpQkFBaUIsQ0FBQyxPQUFPLENBQUMsVUFBVSxFQUFDLE1BQU0sQ0FBQyxDQUFDO0FBQ3RELE9BQU8sQ0FBQyxHQUFHLENBQUMsaUJBQWlCLENBQUMsT0FBTyxDQUFDLFVBQVUsRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLEVBQVksQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUU3RSxVQUFVLENBQUMsY0FBYyxDQUFDLFFBQVEsRUFBRTtJQUNoQyxPQUFPLDhJQUE4SSxDQUFDO0FBQzFKLENBQUMsQ0FBQyxDQUFDOzs7Ozs7Ozs7Ozs7QUN4RUgsZ0NBQWlDO0FBSWpDLHVEQUFvRDtBQUVwRCxNQUFhLFNBQVM7SUFVbEI7UUFUQTs7OztRQUlBO1FBQ2lCLGVBQVUsR0FBZSxFQUFFLENBQUM7UUFLekMsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUM7UUFDbEIsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFDO0lBQ2xDLENBQUM7SUFDRDs7T0FFRztJQUNJLE1BQU0sQ0FBQyxRQUFrQjtRQUM1QixPQUFPLENBQUMsR0FBRyxDQUFDLGdDQUFnQyxDQUFDLENBQUM7UUFDOUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUM7SUFDbkMsQ0FBQztJQUVNLE1BQU0sQ0FBQyxRQUFrQjtRQUM1QixNQUFNLGFBQWEsR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUN4RCxJQUFJLENBQUMsVUFBVSxDQUFDLE1BQU0sQ0FBQyxhQUFhLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDekMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQ0FBZ0MsQ0FBQyxDQUFDO0lBQ2xELENBQUM7SUFFTSxlQUFlLENBQUMsUUFBZ0IsRUFBRSxRQUFnQztRQUNyRSxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxRQUFRLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDOUMsQ0FBQztJQUVEOztPQUVHO0lBQ0ksTUFBTSxDQUFDLFFBQWdCO1FBQzFCLE9BQU8sQ0FBQyxHQUFHLENBQUMsaUNBQWlDLENBQUMsQ0FBQztRQUMvQyxLQUFLLE1BQU0sUUFBUSxJQUFJLElBQUksQ0FBQyxVQUFVLEVBQUU7WUFDcEMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDO1NBQ3BEO0lBQ0wsQ0FBQztJQUVNLGtCQUFrQixDQUFDLGVBQXVDO1FBQzdELElBQUksQ0FBQyxLQUFLLEdBQUcsZUFBZSxDQUFDO1FBQzdCLElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQztRQUNwQixPQUFPLENBQUMsR0FBRyxDQUFDLG9DQUFvQyxDQUFDLENBQUM7UUFDbEQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxlQUFlLENBQUMsQ0FBQztRQUM3QixJQUFJLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDO0lBQzdCLENBQUM7SUFFTSxLQUFLLENBQUMsV0FBVyxDQUFDLFFBQWdCLEVBQUUsVUFBa0IsRUFBRSxlQUE4QjtRQUN6RixNQUFNLElBQUksR0FBRyxNQUFNLGlDQUFlLENBQUMsV0FBVyxDQUFDLFFBQVEsRUFBRSxVQUFVLEVBQUUsZUFBZSxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQzVGLElBQUksSUFBSSxJQUFJLElBQUksRUFBRTtZQUNkLEdBQUcsQ0FBQyxJQUFJLENBQUMsb0NBQW9DLENBQUMsQ0FBQTtZQUM5QyxHQUFHLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDO1lBQ2hCLElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxDQUFDLFFBQVEsRUFBRSxJQUFJLENBQUMsQ0FBQztZQUN0QyxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1NBQ3pCO2FBQ0k7WUFDRCxHQUFHLENBQUMsS0FBSyxDQUFDLG9CQUFvQixDQUFDLENBQUM7U0FDbkM7UUFFRCxPQUFPLElBQUksQ0FBQztJQUNoQixDQUFDO0lBSU8sWUFBWSxLQUFLLENBQUM7SUFFbkIsZ0JBQWdCO0lBRXZCLENBQUM7Q0FJSjtBQTNFRCw4QkEyRUM7Ozs7Ozs7O0FDdEZELGdDQUFnQztBQUVoQyw0RUFBeUU7QUFFekUsa0RBQStDO0FBRS9DLDJEQUF3RDtBQUd4RCw0RUFBeUU7QUFFekUsTUFBYSxlQUFlO0lBR2pCLE1BQU0sQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLFFBQWdCLEVBQUUsVUFBa0IsRUFBRSxlQUE4QixFQUFFLFNBQW9CO1FBQ3RILFFBQVEsZUFBZSxFQUFFO1lBQ3JCLEtBQUssSUFBSSxDQUFDLENBQUM7Z0JBQ1Asb0NBQW9DO2dCQUNwQywwQ0FBMEM7Z0JBQzFDLHVEQUF1RDtnQkFDdkQsNkZBQTZGO2dCQUM3RixxSkFBcUo7Z0JBQ3JKLHVFQUF1RTtnQkFDdkUsa0VBQWtFO2dCQUNsRSxpQ0FBaUM7Z0JBQ2pDLFNBQVM7Z0JBQ1QsU0FBUztnQkFFVCxNQUFNLElBQUksR0FBcUIsTUFBTSxJQUFJLENBQUMsa0JBQWtCLENBQUMsUUFBUSxDQUFDLENBQUM7Z0JBQ3ZFLE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxlQUFlLENBQUMsRUFBRSxFQUFFLFVBQVUsQ0FBQyxDQUFDLENBQUM7YUFFL0Q7WUFDRCxPQUFPLENBQUMsQ0FBQztnQkFDTCxxREFBcUQ7Z0JBQ3JELDBDQUEwQztnQkFDMUMsdURBQXVEO2dCQUN2RCx1RUFBdUU7Z0JBQ3ZFLGtFQUFrRTtnQkFDbEUsZ0NBQWdDO2dCQUNoQyxpQ0FBaUM7Z0JBQ2pDLFNBQVM7Z0JBQ1QsU0FBUztnQkFDVCxNQUFNLElBQUksR0FBcUIsTUFBTSxJQUFJLENBQUMsa0JBQWtCLENBQUMsUUFBUSxFQUFFLGVBQWUsQ0FBQyxDQUFDO2dCQUN4RixPQUFPLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsZUFBZSxDQUFDLEVBQUUsRUFBRSxVQUFVLENBQUMsQ0FBQyxDQUFDO2FBQy9EO1NBQ0o7UUFFRCxlQUFlO0lBQ25CLENBQUM7SUFFTyxNQUFNLENBQUMsZUFBZSxDQUFDLGNBQThCLEVBQUUsVUFBa0I7UUFDN0UsTUFBTSxFQUFFLEdBQUcsSUFBSSwyQ0FBb0IsRUFBRSxDQUFDO1FBQ3RDLEVBQUUsQ0FBQyxRQUFRLEdBQUcsY0FBYyxDQUFDLFFBQVEsQ0FBQztRQUN0QyxFQUFFLENBQUMsTUFBTSxHQUFHLGNBQWMsQ0FBQyxNQUFNLENBQUM7UUFDbEMsRUFBRSxDQUFDLFdBQVcsR0FBRyxjQUFjLENBQUMsV0FBVyxDQUFDO1FBQzVDLEVBQUUsQ0FBQyxPQUFPLEdBQUcsSUFBSSxDQUFDLGlCQUFpQixDQUFDLE9BQU8sQ0FBQyxVQUFVLEVBQUUsY0FBYyxDQUFDLGFBQWEsQ0FBVyxDQUFDO1FBQ2hHLE9BQU8sRUFBRSxDQUFDO0lBQ2QsQ0FBQztJQUVPLE1BQU0sQ0FBQyxLQUFLLENBQUMsa0JBQWtCLENBQUMsTUFBYztRQUNsRCxNQUFNLE9BQU8sR0FBRyxJQUFJLE9BQU8sRUFBRSxDQUFDO1FBQzlCLElBQUksaUJBQU8sQ0FBQyxTQUFTLElBQUksSUFBSSxFQUFFO1lBQzNCLEdBQUcsQ0FBQyxLQUFLLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztZQUM1QixPQUFPO1NBQ1Y7UUFBQSxDQUFDO1FBQ0YsT0FBTyxDQUFDLE1BQU0sQ0FBQyxjQUFjLEVBQUUsaUJBQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUNsRCxNQUFNLFFBQVEsR0FBRyxNQUFNLEtBQUssQ0FBQyxHQUFHLGlCQUFPLENBQUMsaUJBQWlCLElBQUksTUFBTSxFQUFFLEVBQUU7WUFDbkUsTUFBTSxFQUFFLEtBQUs7WUFDYixPQUFPLEVBQUUsT0FBTztTQUNuQixDQUFDLENBQUM7UUFDSCxPQUFPLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDO1FBQzlCLElBQUkscUNBQWlCLENBQUMsUUFBUSxDQUFDLEtBQUssRUFBRSxDQUFDLEVBQUU7WUFDckMsT0FBTyxJQUFJLENBQUM7U0FDZjtRQUNELE1BQU0sSUFBSSxHQUFpQixNQUFNLFFBQVEsQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUNqRCxPQUFPLElBQUksQ0FBQztJQUNoQixDQUFDO0lBRU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxNQUFjLEVBQUUsb0JBQTRCO1FBQ2hGLE1BQU0sT0FBTyxHQUFHLElBQUksT0FBTyxFQUFFLENBQUM7UUFDOUIsSUFBSSxpQkFBTyxDQUFDLFNBQVMsSUFBSSxJQUFJLEVBQUU7WUFDM0IsR0FBRyxDQUFDLEtBQUssQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO1lBQzVCLE9BQU87U0FDVjtRQUFBLENBQUM7UUFDRixPQUFPLENBQUMsTUFBTSxDQUFDLGNBQWMsRUFBRSxpQkFBTyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQ2xELE1BQU0sUUFBUSxHQUFHLE1BQU0sS0FBSyxDQUFDLEdBQUcsaUJBQU8sQ0FBQyxpQkFBaUIsSUFBSSxNQUFNLElBQUksb0JBQW9CLEVBQUUsRUFBRTtZQUMzRixNQUFNLEVBQUUsS0FBSztZQUNiLE9BQU8sRUFBRSxPQUFPO1NBQ25CLENBQUMsQ0FBQztRQUNILE9BQU8sQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUM7UUFDOUIsSUFBSSxxQ0FBaUIsQ0FBQyxRQUFRLENBQUMsS0FBSyxFQUFFLENBQUMsRUFBRTtZQUNyQyxPQUFPLElBQUksQ0FBQztTQUNmO1FBQ0QsTUFBTSxJQUFJLEdBQWlCLE1BQU0sUUFBUSxDQUFDLElBQUksRUFBRSxDQUFDO1FBQ2pELE9BQU8sSUFBSSxDQUFDO0lBQ2hCLENBQUM7O0FBcEZMLDBDQXFGQztBQXBGMkIsaUNBQWlCLEdBQUcsSUFBSSw2Q0FBcUIsRUFBRSxDQUFDOzs7Ozs7OztBQ2Q1RSxnQ0FBaUM7QUFDakMsMkNBQXFDO0FBQ3JDLGtEQUFrRDtBQUVsRCxTQUFnQixpQkFBaUIsQ0FBQyxRQUFrQjtJQUNoRCxrRkFBa0Y7SUFDbEYsSUFBSSxDQUFDLFFBQVEsQ0FBQyxFQUFFLEVBQUU7UUFDZCxPQUFPLFFBQVEsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEVBQUU7WUFDL0Isa0RBQWtEO1lBQ2xELHdDQUF3QztZQUN4QyxpRkFBaUY7WUFDakYsbUZBQW1GO1lBQ25GLEdBQUcsQ0FBQyxLQUFLLENBQUMsb0JBQU8sQ0FBQyxxQ0FBcUMsRUFBRSxRQUFRLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQztZQUMzRSxHQUFHLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDWixPQUFPLElBQUksQ0FBQztRQUNoQixDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEVBQUU7WUFDWCxpREFBaUQ7WUFDakQsdUZBQXVGO1lBQ3ZGLDRFQUE0RTtZQUM1RSxtRkFBbUY7WUFDbkYsR0FBRyxDQUFDLEtBQUssQ0FBQyxvQkFBTyxDQUFDLHFDQUFxQyxFQUFFLFFBQVEsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDO1lBQzNFLEdBQUcsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDaEIsT0FBTyxJQUFJLENBQUM7UUFDaEIsQ0FBQyxDQUFDLENBQUM7S0FDTjtBQUNMLENBQUM7QUFyQkQsOENBcUJDOzs7Ozs7OztBQ3RCRCwyREFBd0Q7QUFFeEQsa0RBQStDO0FBRS9DLGdDQUFnQztBQUVoQyxNQUFhLFNBQVM7SUFRbEIsOENBQThDO0lBRTlDO1FBVEE7Ozs7UUFJQTtRQUNpQixjQUFTLEdBQWUsRUFBRSxDQUFDO0lBSTVCLENBQUM7SUFDakI7O09BRUc7SUFDSSxNQUFNLENBQUMsUUFBa0I7UUFDNUIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxnQ0FBZ0MsQ0FBQyxDQUFDO1FBQzlDLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ2xDLENBQUM7SUFFTSxNQUFNLENBQUMsUUFBa0I7UUFDNUIsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDdkQsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsYUFBYSxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQ3hDLE9BQU8sQ0FBQyxHQUFHLENBQUMsZ0NBQWdDLENBQUMsQ0FBQztJQUNsRCxDQUFDO0lBRUQ7O09BRUc7SUFDSSxNQUFNO1FBQ1QsT0FBTyxDQUFDLEdBQUcsQ0FBQyxpQ0FBaUMsQ0FBQyxDQUFDO1FBQy9DLEtBQUssTUFBTSxRQUFRLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRTtZQUNuQyxRQUFRLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztTQUMvQjtJQUNMLENBQUM7SUFFTSxrQkFBa0IsQ0FBQyxjQUFxQztRQUMzRCxJQUFJLENBQUMsS0FBSyxHQUFHLGNBQWMsQ0FBQztRQUM1QixJQUFJLENBQUMsWUFBWSxFQUFFLENBQUM7UUFDcEIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxvQ0FBb0MsQ0FBQyxDQUFDO1FBQ2xELE9BQU8sQ0FBQyxHQUFHLENBQUMsY0FBYyxDQUFDLENBQUM7UUFDNUIsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDO0lBQ2xCLENBQUM7SUFFRDs7T0FFRztJQUNJLGNBQWM7UUFDakIsSUFBRyxpQkFBTyxDQUFDLFNBQVMsSUFBRyxJQUFJLEVBQUM7WUFDNUIsSUFBSSxDQUFDLGtCQUFrQixDQUFDLGlCQUFPLENBQUMsU0FBUyxFQUFFLGlCQUFPLENBQUMsZ0JBQWdCLENBQUM7aUJBQy9ELElBQUksQ0FBQyxJQUFJLENBQUMsRUFBRTtnQkFDVCx5QkFBeUI7Z0JBQ3pCLCtEQUErRDtnQkFDL0Qsc0RBQXNEO2dCQUN0RCxPQUFPLENBQUMsR0FBRyxDQUFDLHFDQUFxQyxDQUFDLENBQUM7Z0JBQ25ELElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDO2dCQUNsQixJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDbEIsQ0FBQyxDQUFDLENBQUE7U0FDTDthQUNJO1lBQ0QsR0FBRyxDQUFDLEtBQUssQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDO1NBQ25DO0lBQ0wsQ0FBQztJQUVELEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxVQUFrQixFQUFFLEdBQVc7UUFDcEQsSUFBSSxPQUFPLEdBQUcsSUFBSSxPQUFPLEVBQUUsQ0FBQztRQUM1QixtREFBbUQ7UUFDbkQsT0FBTyxDQUFDLE1BQU0sQ0FBQyxjQUFjLEVBQUUsVUFBVSxDQUFDLENBQUM7UUFDM0MsSUFBSSxRQUFRLEdBQUcsTUFBTSxLQUFLLENBQUMsaUJBQU8sQ0FBQyxnQkFBZ0IsRUFBRTtZQUNqRCxNQUFNLEVBQUUsS0FBSztZQUNiLE9BQU8sRUFBRSxPQUFPO1NBQ25CLENBQUMsQ0FBQztRQUNILE9BQU8sQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUM7UUFDOUIsSUFBSSxxQ0FBaUIsQ0FBQyxRQUFRLENBQUMsS0FBSyxFQUFFLENBQUMsRUFBRTtZQUNyQyxPQUFPLElBQUksQ0FBQztTQUNmO1FBQ0QsSUFBSSxJQUFJLEdBQUcsTUFBTSxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDakMsZUFBZTtRQUNmLE9BQU8sSUFBSSxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEVBQUU7WUFDbkMsSUFBSSxJQUFJLElBQUksSUFBSTtnQkFDWixPQUFPLENBQUMsSUFBSSxDQUFDLENBQUE7O2dCQUViLE1BQU0sQ0FBQyxvQkFBb0IsQ0FBQyxDQUFBO1FBQ3BDLENBQUMsQ0FBQyxDQUFBO0lBQ04sQ0FBQztJQUVPLFlBQVksS0FBSyxDQUFDO0NBRTdCO0FBdkZELDhCQXVGQzs7Ozs7Ozs7QUMvRkQsNkJBQTZCO0FBRTdCLE1BQWEscUJBQXFCO0lBQWxDO1FBQ3FCLFdBQU0sR0FBUSxFQUFFLElBQUksRUFBRSxLQUFLLEVBQUUsRUFBRSxFQUFFLEdBQUcsRUFBRSxLQUFLLEVBQUUsRUFBRSxFQUFFLElBQUksRUFBRSxLQUFLLEVBQUMsQ0FBQTtJQVNsRixDQUFDO0lBUlUsT0FBTyxDQUFDLFVBQWtCLEVBQUUsU0FBaUI7UUFDaEQsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDLFVBQVUsRUFBRSxTQUFTLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQzVELENBQUM7SUFFTSxPQUFPLENBQUMsVUFBa0IsRUFBRSxNQUFjO1FBQzdDLDZGQUE2RjtRQUM3RixPQUFPLElBQUksQ0FBQyxPQUFPLENBQUMsVUFBVSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsTUFBTSxDQUFDLEVBQUUsU0FBUyxFQUFFLFNBQVMsQ0FBQyxDQUFDO0lBQ2xGLENBQUM7Q0FDSjtBQVZELHNEQVVDOzs7Ozs7O0FDYkQsSUFBaUIsT0FBTyxDQU92QjtBQVBELFdBQWlCLE9BQU87SUFDcEIsOENBQThDO0lBQ25DLGdCQUFRLEdBQWtCLFlBQVksQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDM0QsaUJBQVMsR0FBa0IsWUFBWSxDQUFDLE9BQU8sQ0FBQyxXQUFXLENBQUMsQ0FBQztJQUMzRCx3QkFBZ0IsR0FBRyw0QkFBNEIsQ0FBQztJQUNoRCx5QkFBaUIsR0FBRyx3QkFBd0IsQ0FBQztBQUU5RCxDQUFDLEVBUGdCLE9BQU8sR0FBUCxlQUFPLEtBQVAsZUFBTyxRQU92Qjs7Ozs7QUNQRCxNQUFhLGVBQWU7SUFDeEIsK0VBQStFO0lBRS9FLDhCQUE4QjtJQUM5QixpREFBaUQ7SUFDakQsaUdBQWlHO0lBQ2pHLHdFQUF3RTtJQUN4RSw4Q0FBOEM7SUFDOUMsd0RBQXdEO0lBQ3hELFlBQVk7SUFDWixnREFBZ0Q7SUFDaEQsZ0dBQWdHO0lBQ2hHLHdFQUF3RTtJQUN4RSw4Q0FBOEM7SUFDOUMsd0RBQXdEO0lBQ3hELFlBQVk7SUFDWiwyQ0FBMkM7SUFDM0MsZ0dBQWdHO0lBQ2hHLHdFQUF3RTtJQUN4RSw4Q0FBOEM7SUFDOUMsd0RBQXdEO0lBQ3hELFlBQVk7SUFDWixtQkFBbUI7SUFDbkIsd0RBQXdEO0lBQ3hELFFBQVE7SUFDUixJQUFJO0lBRUosTUFBTSxDQUFDLFdBQVcsQ0FBQyxZQUFvQjtRQUNuQyx3Q0FBd0M7UUFDeEMsSUFBSSxNQUFNLEdBQUcsUUFBUSxDQUFDLGNBQWMsQ0FBQyxZQUFZLENBQUMsQ0FBQyxTQUFTLENBQUM7UUFDN0QsSUFBSSxvQkFBb0IsR0FBRyxVQUFVLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQ3RELE9BQU8sb0JBQW9CLENBQUM7SUFDaEMsQ0FBQztDQUNKO0FBakNELDBDQWlDQzs7Ozs7Ozs7OztBQzVCRCxpRUFBOEQ7QUFJOUQsZ0NBQWdDO0FBQ2hDLHVDQUF1QztBQUN2QywwQ0FBMkM7QUFDM0MsSUFBSSxFQUFFLEdBQUcsSUFBSSxVQUFVLEVBQUUsQ0FBQztBQUUxQixNQUFhLFFBQVE7SUFNakIsWUFBWSxLQUFnQixFQUFFLE9BQW9CO1FBSGpDLHlCQUFvQixHQUFHLGlDQUFlLENBQUMsV0FBVyxDQUFDLDZCQUE2QixDQUFDLENBQUM7UUFJL0YsSUFBSSxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUM7UUFDcEIsSUFBSSxDQUFDLFFBQVEsR0FBRyxPQUFPLENBQUM7SUFDNUIsQ0FBQztJQUdELE1BQU0sQ0FBQyxJQUE0QjtRQUMvQixHQUFHLENBQUMsSUFBSSxDQUFDLHlCQUF5QixDQUFDLENBQUM7UUFDcEMsSUFBSSxJQUFJLEdBQVcsRUFBRSxDQUFDO1FBQ3RCLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQyxFQUF3QixFQUFFLEVBQUU7WUFDdEMsSUFBSSxJQUFJLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxFQUFFLENBQUMsQ0FBQztRQUMxQyxDQUFDLENBQUMsQ0FBQztRQUVIOztXQUVHO1FBQ0gsSUFBSSxHQUFHLFNBQVMsQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO1FBQzNDLElBQUksQ0FBQyxRQUFRLENBQUMsU0FBUyxHQUFHLElBQUksQ0FBQztRQUMvQixHQUFHLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDdkMsQ0FBQztDQUNKO0FBMUJELDRCQTBCQzs7Ozs7Ozs7QUNuQ0QsaUVBQThEO0FBRzlELGdDQUFpQztBQUVqQyxNQUFhLFFBQVE7SUFJakIsa0NBQWtDO0lBR2xDLFlBQVksS0FBWSxFQUFFLFNBQW9CLEVBQUUsT0FBb0I7UUFDaEUsSUFBSSxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUM7UUFDcEIsSUFBSSxDQUFDLFVBQVUsR0FBRyxTQUFTLENBQUM7UUFDNUIsSUFBSSxDQUFDLFFBQVEsR0FBRyxPQUFPLENBQUM7SUFDNUIsQ0FBQztJQUtELE1BQU0sQ0FBQyxJQUEyQjtRQUM5QixJQUFJLFFBQVEsR0FBRyxpQ0FBZSxDQUFDLFdBQVcsQ0FBQyw4QkFBOEIsQ0FBQyxDQUFDO1FBQzNFLElBQUksSUFBSSxHQUFXLEVBQUUsQ0FBQztRQUN0QixJQUFJLENBQUMsT0FBTyxDQUFDLENBQUMsT0FBNEIsRUFBRSxFQUFFO1lBQzFDLElBQUksSUFBSSxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDOUIsQ0FBQyxDQUFDLENBQUM7UUFDSCxJQUFJLENBQUMsUUFBUSxDQUFDLFNBQVMsR0FBRyxJQUFJLENBQUM7UUFDL0IsSUFBSSxDQUFDLGdCQUFnQixFQUFFLENBQUM7UUFDeEIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQ3pDLENBQUM7SUFFTyxNQUFNO0lBRWQsQ0FBQztJQUVPLGdCQUFnQjtRQUNwQixJQUFJLFNBQVMsR0FBRyxRQUFRLENBQUMsc0JBQXNCLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDNUQsS0FBSyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLFNBQVMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxFQUFFLEVBQUU7WUFDdkMsSUFBSSxPQUFPLEdBQUcsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQzNCLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxnQkFBZ0IsQ0FBQyxPQUFPLEVBQUUsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFDLENBQUM7U0FDakY7SUFDTCxDQUFDO0lBR08sWUFBWSxDQUFDLEVBQVc7UUFDNUIsSUFBSSxPQUFPLEdBQUcsUUFBUSxDQUFDLHNCQUFzQixDQUFDLGlCQUFpQixDQUFDLENBQUM7UUFFakUsSUFBSSxVQUFVLEdBQVcsRUFBRSxDQUFDO1FBQzVCLElBQUksT0FBTyxDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUU7WUFDcEIsSUFBSSxlQUFlLEdBQUcsUUFBUSxDQUFDLGNBQWMsQ0FBQyxZQUFZLENBQVEsQ0FBQztZQUVuRSxJQUFJLGVBQWUsSUFBSSxJQUFJLEVBQUU7Z0JBQ3pCLEdBQUcsQ0FBQyxLQUFLLENBQUMsMkNBQTJDLENBQUMsQ0FBQztnQkFDdkQsT0FBTzthQUNWO1lBQ0QsVUFBVSxHQUFHLGVBQWUsQ0FBQyxLQUFLLENBQUE7WUFDbEMsSUFBSSxVQUFVLElBQUksRUFBRSxJQUFJLFVBQVUsSUFBSSxJQUFJLEVBQUU7Z0JBQ3hDLG1DQUFtQztnQkFDbkMsK0NBQStDO2dCQUMvQyxHQUFHLENBQUMsS0FBSyxDQUFDLDZCQUE2QixDQUFDLENBQUM7Z0JBQ3pDLE9BQU87YUFDVjtZQUNELE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxTQUFTLEdBQUcsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUMsU0FBUyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1NBRXRFO1FBQ0QscURBQXFEO2FBQ2hELElBQUksT0FBTyxDQUFDLE1BQU0sSUFBSSxDQUFDLEVBQUU7WUFDMUIsSUFBSSxJQUFJLEdBQUcsUUFBUSxDQUFDLGNBQWMsQ0FBQyxvQkFBb0IsQ0FBUSxDQUFDO1lBQ2hFLElBQUcsSUFBSSxJQUFJLElBQUksRUFDZjtnQkFDSSxHQUFHLENBQUMsS0FBSyxDQUFDLDJDQUEyQyxDQUFDLENBQUM7Z0JBQ3ZELE9BQU87YUFDVjtZQUNELFVBQVUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDO1lBQ3hCLElBQUksVUFBVSxJQUFJLEVBQUUsSUFBSSxVQUFVLElBQUksSUFBSSxFQUFFO2dCQUN4QywwQ0FBMEM7Z0JBQzFDLHNEQUFzRDtnQkFDdEQsR0FBRyxDQUFDLEtBQUssQ0FBQyw2QkFBNkIsQ0FBQyxDQUFDO2dCQUN6QyxPQUFPO2FBQ1Y7WUFDRCx5Q0FBeUM7WUFDekMsUUFBUSxDQUFDLGNBQWMsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUM7WUFDMUQseUNBQXlDO1lBQ3pDLFFBQVEsQ0FBQyxjQUFjLENBQUMsV0FBVyxDQUFDLENBQUMsTUFBTSxHQUFHLEtBQUssQ0FBQztZQUNwRCx5Q0FBeUM7WUFDekMsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFJLENBQUM7U0FDdEI7UUFDRCw0REFBNEQ7UUFDNUQsSUFBSSxJQUFJLEdBQUcsRUFBRSxDQUFDLHNCQUFzQixDQUFDLGNBQWMsQ0FBQyxDQUFDLENBQUMsQ0FBZ0IsQ0FBQztRQUN2RSxJQUFJLFFBQVEsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDO1FBQzlCLHlDQUF5QztRQUN6QyxRQUFRLENBQUMsY0FBYyxDQUFDLGdCQUFnQixDQUFDLENBQUMsU0FBUyxHQUFHLFFBQVEsQ0FBQztRQUMvRCxJQUFJLENBQUMsVUFBVSxDQUFDLFdBQVcsQ0FBQyxRQUFRLEVBQUUsVUFBVSxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ3hELDBDQUEwQztRQUMxQyxjQUFjLENBQUMsT0FBTyxDQUFDLGNBQWMsRUFBRSxRQUFRLENBQUMsQ0FBQztRQUNqRCxFQUFFLENBQUMsU0FBUyxJQUFJLFNBQVMsQ0FBQztJQUM5QixDQUFDO0NBRUo7QUE5RkQsNEJBOEZDOzs7Ozs7O0FDeEdELE1BQWEsbUJBQW1CO0NBSS9CO0FBSkQsa0RBSUM7Ozs7O0FDSkQsTUFBYSxvQkFBb0I7Q0FPaEM7QUFQRCxvREFPQyIsImZpbGUiOiJnZW5lcmF0ZWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uKCl7ZnVuY3Rpb24gcihlLG4sdCl7ZnVuY3Rpb24gbyhpLGYpe2lmKCFuW2ldKXtpZighZVtpXSl7dmFyIGM9XCJmdW5jdGlvblwiPT10eXBlb2YgcmVxdWlyZSYmcmVxdWlyZTtpZighZiYmYylyZXR1cm4gYyhpLCEwKTtpZih1KXJldHVybiB1KGksITApO3ZhciBhPW5ldyBFcnJvcihcIkNhbm5vdCBmaW5kIG1vZHVsZSAnXCIraStcIidcIik7dGhyb3cgYS5jb2RlPVwiTU9EVUxFX05PVF9GT1VORFwiLGF9dmFyIHA9bltpXT17ZXhwb3J0czp7fX07ZVtpXVswXS5jYWxsKHAuZXhwb3J0cyxmdW5jdGlvbihyKXt2YXIgbj1lW2ldWzFdW3JdO3JldHVybiBvKG58fHIpfSxwLHAuZXhwb3J0cyxyLGUsbix0KX1yZXR1cm4gbltpXS5leHBvcnRzfWZvcih2YXIgdT1cImZ1bmN0aW9uXCI9PXR5cGVvZiByZXF1aXJlJiZyZXF1aXJlLGk9MDtpPHQubGVuZ3RoO2krKylvKHRbaV0pO3JldHVybiBvfXJldHVybiByfSkoKSIsImltcG9ydCB7IENvbnRyb2xsZXIgfSBmcm9tIFwiLi9BYnN0cmFjdENvbnRyb2xsZXJcIjtcbmltcG9ydCBcIi4uL21vZGVsL0Fic3RyYWN0TW9kZWxcIlxuaW1wb3J0IFwiLi4vbW9kZWwvVXNlck1vZGVsXCJcbmltcG9ydCBcIi4uL3ZpZXcvQWJzdHJhY3RWaWV3XCJcbmltcG9ydCBcIi4uL3ZpZXcvVXNlclZpZXdcIlxuaW1wb3J0IHsgTW9kZWwgfSBmcm9tIFwiLi4vbW9kZWwvQWJzdHJhY3RNb2RlbFwiO1xuaW1wb3J0IHsgVmlldyB9IGZyb20gXCIuLi92aWV3L0Fic3RyYWN0Vmlld1wiO1xuaW1wb3J0IHsgQ2hhdE1lc3NhZ2VWaWV3TW9kZWwgfSBmcm9tIFwiLi4vdmlld21vZGVsL0NoYXRNZXNzYWdlVmlld01vZGVsXCI7XG5pbXBvcnQgeyBDaGF0TW9kZWwgfSBmcm9tIFwiLi4vbW9kZWwvQ2hhdE1vZGVsXCI7XG5pbXBvcnQgeyBDaGF0VmlldyB9IGZyb20gXCIuLi92aWV3L0NoYXRWaWV3XCI7XG5cbmV4cG9ydCBjbGFzcyBDaGF0Q29udHJvbGxlciB7XG4gICAgcHJpdmF0ZSBfbW9kZWw6IENoYXRNb2RlbDtcbiAgICBwcml2YXRlIF92aWV3OiBDaGF0VmlldztcblxuXG4gICAgY29uc3RydWN0b3IobW9kZWw6IENoYXRNb2RlbCwgdmlldzogQ2hhdFZpZXcpIHtcbiAgICAgICAgdGhpcy5fbW9kZWwgPSBtb2RlbDtcbiAgICAgICAgdGhpcy5fdmlldyA9IHZpZXc7XG4gICAgfVxuXG5cbiAgICAvKipcbiAgICAgKiBldmVudEhhbmRsZXJcbiAgICAgKi9cbiAgICBwdWJsaWMgZXZlbnRIYW5kbGVyKHZtOiBDaGF0TWVzc2FnZVZpZXdNb2RlbFtdKTogdm9pZCB7XG4gICAgICAgIHRoaXMuX21vZGVsLnNvbWVCdXNpbmVzc01ldGhvZCh2bSk7XG4gICAgfVxuXG4gICAgcHVibGljIHRlc3QoKTogdm9pZCB7XG4gICAgICAgIGNvbnN0IGNoYXRNZXNzYWdlVmlld01vZGVsczogQ2hhdE1lc3NhZ2VWaWV3TW9kZWxbXSA9IFtdO1xuICAgICAgICBsZXQgY2hhdE1lc3NhZ2VWaWV3TW9kZWxNb2NrID0gbmV3IENoYXRNZXNzYWdlVmlld01vZGVsKCk7XG4gICAgICAgIGNoYXRNZXNzYWdlVmlld01vZGVsTW9jay5mcm9tVXNlciA9IFwidXNlcjFcIjtcbiAgICAgICAgY2hhdE1lc3NhZ2VWaWV3TW9kZWxNb2NrLnRvVXNlciA9IFwidXNlcjJcIjtcbiAgICAgICAgY2hhdE1lc3NhZ2VWaWV3TW9kZWxNb2NrLm1lc3NhZ2UgPSBcIlwiO1xuICAgICAgICBjaGF0TWVzc2FnZVZpZXdNb2RlbE1vY2subWVzc2FnZVRpbWUgPSBuZXcgRGF0ZSgpO1xuICAgICAgICBjaGF0TWVzc2FnZVZpZXdNb2RlbHMucHVzaChjaGF0TWVzc2FnZVZpZXdNb2RlbE1vY2spO1xuICAgIH1cblxuICAgIFxufSIsImltcG9ydCB7IENvbnRyb2xsZXIgfSBmcm9tIFwiLi9BYnN0cmFjdENvbnRyb2xsZXJcIjtcbmltcG9ydCBcIi4uL21vZGVsL0Fic3RyYWN0TW9kZWxcIlxuaW1wb3J0IFwiLi4vbW9kZWwvVXNlck1vZGVsXCJcbmltcG9ydCBcIi4uL3ZpZXcvQWJzdHJhY3RWaWV3XCJcbmltcG9ydCBcIi4uL3ZpZXcvVXNlclZpZXdcIlxuaW1wb3J0IHsgTW9kZWwgfSBmcm9tIFwiLi4vbW9kZWwvQWJzdHJhY3RNb2RlbFwiO1xuaW1wb3J0IHsgVmlldyB9IGZyb20gXCIuLi92aWV3L0Fic3RyYWN0Vmlld1wiO1xuaW1wb3J0IHsgQWN0aXZlVXNlclZpZXdNb2RlbCB9IGZyb20gXCIuLi92aWV3bW9kZWwvQWN0aXZlVXNlclZpZXdNb2RlbFwiO1xuaW1wb3J0IHsgVXNlclZpZXcgfSBmcm9tIFwiLi4vdmlldy9Vc2VyVmlld1wiO1xuaW1wb3J0IHsgVXNlck1vZGVsIH0gZnJvbSBcIi4uL21vZGVsL1VzZXJNb2RlbFwiO1xuXG5leHBvcnQgY2xhc3MgVXNlckNvbnRyb2xsZXIge1xuICAgIHByaXZhdGUgX21vZGVsOiBVc2VyTW9kZWw7XG4gICAgcHJpdmF0ZSBfdmlldzogVXNlclZpZXc7XG5cblxuICAgIGNvbnN0cnVjdG9yKG1vZGVsOiBVc2VyTW9kZWwsIHZpZXc6IFVzZXJWaWV3KSB7XG4gICAgICAgIHRoaXMuX21vZGVsID0gbW9kZWw7XG4gICAgICAgIHRoaXMuX3ZpZXcgPSB2aWV3O1xuICAgIH1cblxuXG4gICAgLyoqXG4gICAgICogZXZlbnRIYW5kbGVyXG4gICAgICovXG4gICAgcHVibGljIGV2ZW50SGFuZGxlcih2bTogQWN0aXZlVXNlclZpZXdNb2RlbFtdKTogdm9pZCB7XG4gICAgICAgIHRoaXMuX21vZGVsLnNvbWVCdXNpbmVzc01ldGhvZCh2bSk7XG4gICAgfVxuXG4gICAgcHVibGljIHRlc3QoKTogdm9pZCB7XG4gICAgICAgIGNvbnN0IGFjdGl2ZVVzZXJzTW9jazogQWN0aXZlVXNlclZpZXdNb2RlbFtdID0gW107XG4gICAgICAgIGxldCBhY3RpdmVVc2VyVmlld01vZGVsTW9jayA9IG5ldyBBY3RpdmVVc2VyVmlld01vZGVsKCk7XG4gICAgICAgIGFjdGl2ZVVzZXJWaWV3TW9kZWxNb2NrLnVzZXJOYW1lID0gXCJzb21lIHVzZXJcIjtcbiAgICAgICAgYWN0aXZlVXNlclZpZXdNb2RlbE1vY2subGFzdEFjdGl2ZSA9IFwiMyBocnMgYWdvXCI7XG4gICAgICAgIGFjdGl2ZVVzZXJWaWV3TW9kZWxNb2NrLm9ubGluZSA9IHRydWU7XG4gICAgICAgIGFjdGl2ZVVzZXJzTW9jay5wdXNoKGFjdGl2ZVVzZXJWaWV3TW9kZWxNb2NrKTtcbiAgICAgICAgYWN0aXZlVXNlclZpZXdNb2RlbE1vY2sgPSBuZXcgQWN0aXZlVXNlclZpZXdNb2RlbCgpO1xuICAgICAgICBhY3RpdmVVc2VyVmlld01vZGVsTW9jay5sYXN0QWN0aXZlID0gXCIzIGhycyBhZ29cIjtcbiAgICAgICAgYWN0aXZlVXNlclZpZXdNb2RlbE1vY2sub25saW5lID0gdHJ1ZTtcbiAgICAgICAgYWN0aXZlVXNlclZpZXdNb2RlbE1vY2sudXNlck5hbWUgPSBcInNvbWUgdXNlciAyXCI7XG4gICAgICAgIGFjdGl2ZVVzZXJzTW9jay5wdXNoKGFjdGl2ZVVzZXJWaWV3TW9kZWxNb2NrKTtcbiAgICAgICAgdGhpcy5ldmVudEhhbmRsZXIoYWN0aXZlVXNlcnNNb2NrKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0QWN0aXZlVXNlcnMoKTogdm9pZCB7XG4gICAgICAgIHRoaXMuX21vZGVsLmdldEFjdGl2ZVVzZXJzKCk7XG4gICAgfVxuXG4gICAgXG59IiwiaW1wb3J0IHsgQ29udHJvbGxlciB9IGZyb20gXCIuL2NvbnRyb2xsZXIvQWJzdHJhY3RDb250cm9sbGVyXCI7XG5pbXBvcnQgeyBVc2VyTW9kZWwgfSBmcm9tIFwiLi9tb2RlbC9Vc2VyTW9kZWxcIlxuaW1wb3J0IHsgTW9kZWwgfSBmcm9tIFwiLi9tb2RlbC9BYnN0cmFjdE1vZGVsXCI7XG5pbXBvcnQgeyBWaWV3IH0gZnJvbSBcIi4vdmlldy9BYnN0cmFjdFZpZXdcIjtcbmltcG9ydCB7IFVzZXJWaWV3IH0gZnJvbSBcIi4vdmlldy9Vc2VyVmlld1wiO1xuaW1wb3J0IHsgVXNlckNvbnRyb2xsZXIgfSBmcm9tIFwiLi9jb250cm9sbGVyL1VzZXJDb250cm9sbGVyXCI7XG5pbXBvcnQgeyBNb2RlbEZhY3RvcnkgfSBmcm9tIFwiLi9tb2RlbC9Nb2RlbEZhY3RvcnlcIjtcbmltcG9ydCB7IEFjdGl2ZVVzZXJWaWV3TW9kZWwgfSBmcm9tIFwiLi92aWV3bW9kZWwvQWN0aXZlVXNlclZpZXdNb2RlbFwiO1xuaW1wb3J0IHsgQ2hhdE1lc3NhZ2VWaWV3TW9kZWwgfSBmcm9tIFwiLi92aWV3bW9kZWwvQ2hhdE1lc3NhZ2VWaWV3TW9kZWxcIjtcbmltcG9ydCAqIGFzIEhhbmRsZWJhcnMgZnJvbSBcImhhbmRsZWJhcnNcIjtcbmltcG9ydCBtYXJrZG93bml0ID0gcmVxdWlyZSgnbWFya2Rvd24taXQnKTtcbmltcG9ydCB7IENoYXRNb2RlbCB9IGZyb20gXCIuL21vZGVsL0NoYXRNb2RlbFwiO1xuaW1wb3J0IHsgQ2hhdFZpZXcgfSBmcm9tIFwiLi92aWV3L0NoYXRWaWV3XCI7XG5pbXBvcnQgeyBDaGF0Q29udHJvbGxlciB9IGZyb20gXCIuL2NvbnRyb2xsZXIvQ2hhdENvbnRyb2xsZXJcIjtcbmltcG9ydCB7IEpzb25BUEkgfSBmcm9tIFwiLi9zaW5nbGV0b24vSnNvbkFQSVwiO1xuLy8gaW1wb3J0IGxvZyA9IHJlcXVpcmUoJ2xvZ2xldmVsJylcbmltcG9ydCAqIGFzIGxvZyBmcm9tICdsb2dsZXZlbCc7XG4vLyBpbXBvcnQgbG9nIGZyb20gJ2xvZ2xldmVsJztcbmltcG9ydCB7IEVuY3J5cHRpb25TZXJ2aWNlIH0gZnJvbSBcIi4vc2VydmljZS9FbmNyeXB0aW9uU2VydmljZVwiO1xuaW1wb3J0IHsgU0pDTEVuY3J5cHRpb25TZXJ2aWNlIH0gZnJvbSBcIi4vc2VydmljZS9TSkNMRW5jcnlwdGlvblNlcnZpY2VcIjtcbi8vIHZhciBtYXJrZG93bml0ID0gcmVxdWlyZSgnbWFya2Rvd24taXQnKTtcbnZhciBtZCA9IG5ldyBtYXJrZG93bml0KCk7XG5cblxuY29uc3QgdXNlckJveCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdjb250YWN0cy1ib3gnKTtcblxubG9nLnNldExldmVsKFwiVFJBQ0VcIilcblxuY29uc3QgY2hhdE1vZGVsID0gbmV3IENoYXRNb2RlbCgpO1xuXG5jb25zdCB1c2VyTW9kZWwgPSBuZXcgVXNlck1vZGVsKCk7XG4vLyBjb25zdCB1c2VyTW9kZWwgPSBNb2RlbEZhY3RvcnkuY3JlYXRlTW9kZWwoXCJVU0VSXCIpO1xuLy8gQHRzLWlnbm9yZTogQXJndW1lbnQgb2YgdHlwZSAnSFRNTEVsZW1lbnQgfCBudWxsJyBpcyBub3QgYXNzaWduYWJsZSB0byBwYXJhbWV0ZXIgb2YgdHlwZSAnSFRNTEVsZW1lbnQnLiBUeXBlICdudWxsJyBpcyBub3QgYXNzaWduYWJsZSB0byB0eXBlICdIVE1MRWxlbWVudCcuXG5jb25zdCB1c2VyVmlldyA9IG5ldyBVc2VyVmlldyh1c2VyTW9kZWwsIGNoYXRNb2RlbCwgdXNlckJveCk7XG5cbi8vIGNvbnNvbGUubG9nKHVzZXJCb3gpO1xuXG51c2VyTW9kZWwuYXR0YWNoKHVzZXJWaWV3KTtcbi8vIHVzZXJWaWV3Lm1vZGVsXG5cblxuY29uc3QgdXNlckNvbnRyb2xsZXIgPSBuZXcgVXNlckNvbnRyb2xsZXIodXNlck1vZGVsLCB1c2VyVmlldyk7XG4vLyB1c2VyQ29udHJvbGxlci50ZXN0KCk7XG5cblxuXG4vLyB1c2VyTW9kZWwuc29tZUJ1c2luZXNzTWV0aG9kKGFjdGl2ZVVzZXJzTW9jayk7XG5sb2cuaW5mbyhcImhlbGxvXCIpO1xuXG5jb25zdCBjaGF0QXJlYSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdjaGF0LWFyZWEtbmV3Jyk7XG4vLyBAdHMtaWdub3JlOiBBcmd1bWVudCBvZiB0eXBlICdIVE1MRWxlbWVudCB8IG51bGwnIGlzIG5vdCBhc3NpZ25hYmxlIHRvIHBhcmFtZXRlciBvZiB0eXBlICdIVE1MRWxlbWVudCcuIFR5cGUgJ251bGwnIGlzIG5vdCBhc3NpZ25hYmxlIHRvIHR5cGUgJ0hUTUxFbGVtZW50Jy5cbmNvbnN0IGNoYXRWaWV3ID0gbmV3IENoYXRWaWV3KGNoYXRNb2RlbCwgY2hhdEFyZWEpO1xuY2hhdE1vZGVsLmF0dGFjaChjaGF0Vmlldyk7XG5jb25zdCBjaGF0Q29udHJvbGxlciA9IG5ldyBDaGF0Q29udHJvbGxlcihjaGF0TW9kZWwsIGNoYXRWaWV3KTtcblxuXG5mdW5jdGlvbiBzb21lRnVuYyh2bTogQWN0aXZlVXNlclZpZXdNb2RlbCk6IHZvaWQge1xuICAgIC8vIGxvZy5pbmZvKHZtKTtcbiAgICAvLyBsb2dnZXIuaW5mbyh2bSlcbn1cblxudXNlckNvbnRyb2xsZXIuZ2V0QWN0aXZlVXNlcnMoKTtcblxubG9nLmluZm8oXCJ0ZXN0XCIpO1xuLy8gc29tZUZ1bmMoYWN0aXZlVXNlclZpZXdNb2RlbE1vY2spO1xuXG4vLyBAdHMtaWdub3JlOiBPYmplY3QgaXMgcG9zc2libHkgJ251bGwnLlxudmFyIHNvdXJjZSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKFwibXNnX2NvbnRhaW5lcl90ZW1wbGF0ZVwiKS5pbm5lckhUTUw7XG5cbnZhciBtc2dDb250YWluZXJUZW1wbGF0ZSA9IEhhbmRsZWJhcnMuY29tcGlsZShzb3VyY2UpO1xuXG5Kc29uQVBJLkFDVElWRV9VU0VSU19HRVQgKyAnYWVmJztcblxuY29uc3QgZW5jcnlwdGlvblNlcnZpY2U6IEVuY3J5cHRpb25TZXJ2aWNlID0gbmV3IFNKQ0xFbmNyeXB0aW9uU2VydmljZSgpO1xubGV0IGN0ID0gZW5jcnlwdGlvblNlcnZpY2UuZW5jcnlwdChcInBhc3N3b3JkXCIsXCJkYXRhXCIpO1xuY29uc29sZS5sb2coZW5jcnlwdGlvblNlcnZpY2UuZGVjcnlwdChcInBhc3N3b3JkXCIsIEpTT04ucGFyc2UoY3QgYXMgc3RyaW5nKSkpO1xuXG5IYW5kbGViYXJzLnJlZ2lzdGVySGVscGVyKCdhdmF0YXInLCBmdW5jdGlvbigpIHtcbiAgICByZXR1cm4gJzxkaXYgY2xhc3M9XCJpbWdfY29udF9tc2dcIj4gPGltZyBzcmM9XCJodHRwczovL3N0YXRpYy50dXJib3NxdWlkLmNvbS9QcmV2aWV3LzAwMTI5Mi80ODEvV1YvX0QuanBnXCIgY2xhc3M9XCJyb3VuZGVkLWNpcmNsZSB1c2VyX2ltZ19tc2dcIj4gPC9kaXY+Jztcbn0pOyIsImltcG9ydCB7IFN1YmplY3QgfSBmcm9tIFwiLi4vb2JzZXJ2ZS9PYnNlcnZhYmxlXCI7XG5pbXBvcnQgeyBNb2RlbCB9IGZyb20gXCIuL0Fic3RyYWN0TW9kZWxcIjtcbmltcG9ydCB7IE9ic2VydmVyIH0gZnJvbSBcIi4uL29ic2VydmUvT2JzZXJ2ZXJcIjtcbmltcG9ydCB7IGZldGNoRXJyb3JIYW5kbGVyIH0gZnJvbSBcIi4vRmV0Y2hFcnJvckhhbmRsZXJcIjtcbmltcG9ydCB7IEFjdGl2ZVVzZXJWaWV3TW9kZWwgfSBmcm9tIFwiLi4vdmlld21vZGVsL0FjdGl2ZVVzZXJWaWV3TW9kZWxcIjtcbmltcG9ydCB7IENoYXRNZXNzYWdlVmlld01vZGVsIH0gZnJvbSBcIi4uL3ZpZXdtb2RlbC9DaGF0TWVzc2FnZVZpZXdNb2RlbFwiO1xuaW1wb3J0IHsgSnNvbkFQSSB9IGZyb20gXCIuLi9zaW5nbGV0b24vSnNvbkFQSVwiO1xuaW1wb3J0IGxvZyA9IHJlcXVpcmUoJ2xvZ2xldmVsJyk7XG5pbXBvcnQgeyBFbmNyeXB0aW9uU2VydmljZSB9IGZyb20gXCIuLi9zZXJ2aWNlL0VuY3J5cHRpb25TZXJ2aWNlXCI7XG5pbXBvcnQgeyBTSkNMRW5jcnlwdGlvblNlcnZpY2UgfSBmcm9tIFwiLi4vc2VydmljZS9TSkNMRW5jcnlwdGlvblNlcnZpY2VcIjtcbmltcG9ydCB7IENoYXRNZXNzYWdlRFRPIH0gZnJvbSBcIi4uL2R0by9DaGF0TWVzc2FnZURUT1wiO1xuaW1wb3J0IHsgQ2hhdE1vZGVsSGVscGVyIH0gZnJvbSBcIi4vQ2hhdE1vZGVsSGVscGVyXCI7XG5cbmV4cG9ydCBjbGFzcyBDaGF0TW9kZWwgaW1wbGVtZW50cyBTdWJqZWN0IHtcbiAgICAvKipcbiAgKiBAdHlwZSB7T2JzZXJ2ZXJbXX0gTGlzdCBvZiBzdWJzY3JpYmVycy4gSW4gcmVhbCBsaWZlLCB0aGUgbGlzdCBvZlxuICAqIHN1YnNjcmliZXJzIGNhbiBiZSBzdG9yZWQgbW9yZSBjb21wcmVoZW5zaXZlbHkgKGNhdGVnb3JpemVkIGJ5IGV2ZW50XG4gICogdHlwZSwgZXRjLikuXG4gICovXG4gICAgcHJpdmF0ZSByZWFkb25seSBfb2JzZXJ2ZXJzOiBPYnNlcnZlcltdID0gW107XG4gICAgcHJpdmF0ZSBzdGF0ZTogQ2hhdE1lc3NhZ2VWaWV3TW9kZWxbXSB8IG51bGw7XG4gICAgcHJpdmF0ZSByZWFkb25seSBfbWVzc2FnZXNNYXA6IE1hcDxzdHJpbmcsIENoYXRNZXNzYWdlVmlld01vZGVsW10+O1xuXG4gICAgY29uc3RydWN0b3IoKSB7XG4gICAgICAgIHRoaXMuc3RhdGUgPSBudWxsO1xuICAgICAgICB0aGlzLl9tZXNzYWdlc01hcCA9IG5ldyBNYXAoKTtcbiAgICB9XG4gICAgLyoqXG4gICAgICogVGhlIHN1YnNjcmlwdGlvbiBtYW5hZ2VtZW50IG1ldGhvZHMuXG4gICAgICovXG4gICAgcHVibGljIGF0dGFjaChvYnNlcnZlcjogT2JzZXJ2ZXIpOiB2b2lkIHtcbiAgICAgICAgY29uc29sZS5sb2coJ1N1YmplY3Q6IEF0dGFjaGVkIGFuIG9ic2VydmVyLicpO1xuICAgICAgICB0aGlzLl9vYnNlcnZlcnMucHVzaChvYnNlcnZlcik7XG4gICAgfVxuXG4gICAgcHVibGljIGRldGFjaChvYnNlcnZlcjogT2JzZXJ2ZXIpOiB2b2lkIHtcbiAgICAgICAgY29uc3Qgb2JzZXJ2ZXJJbmRleCA9IHRoaXMuX29ic2VydmVycy5pbmRleE9mKG9ic2VydmVyKTtcbiAgICAgICAgdGhpcy5fb2JzZXJ2ZXJzLnNwbGljZShvYnNlcnZlckluZGV4LCAxKTtcbiAgICAgICAgY29uc29sZS5sb2coJ1N1YmplY3Q6IERldGFjaGVkIGFuIG9ic2VydmVyLicpO1xuICAgIH1cblxuICAgIHB1YmxpYyBzZXRVc2VyTWVzc2FnZXModXNlcm5hbWU6IHN0cmluZywgbWVzc2FnZXM6IENoYXRNZXNzYWdlVmlld01vZGVsW10pIHtcbiAgICAgICAgdGhpcy5fbWVzc2FnZXNNYXAuc2V0KHVzZXJuYW1lLCBtZXNzYWdlcyk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogVHJpZ2dlciBhbiB1cGRhdGUgaW4gZWFjaCBzdWJzY3JpYmVyLlxuICAgICAqL1xuICAgIHB1YmxpYyBub3RpZnkodXNlck5hbWU6IHN0cmluZyk6IHZvaWQge1xuICAgICAgICBjb25zb2xlLmxvZygnU3ViamVjdDogTm90aWZ5aW5nIG9ic2VydmVycy4uLicpO1xuICAgICAgICBmb3IgKGNvbnN0IG9ic2VydmVyIG9mIHRoaXMuX29ic2VydmVycykge1xuICAgICAgICAgICAgb2JzZXJ2ZXIudXBkYXRlKHRoaXMuX21lc3NhZ2VzTWFwLmdldCh1c2VyTmFtZSkpO1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgcHVibGljIHNvbWVCdXNpbmVzc01ldGhvZChjaGF0TWVzc2FnZUxpc3Q6IENoYXRNZXNzYWdlVmlld01vZGVsW10pOiB2b2lkIHtcbiAgICAgICAgdGhpcy5zdGF0ZSA9IGNoYXRNZXNzYWdlTGlzdDtcbiAgICAgICAgdGhpcy5oZWxwZXJNZXRob2QoKTtcbiAgICAgICAgY29uc29sZS5sb2coYFN1YmplY3Q6IE15IHN0YXRlIGhhcyBqdXN0IGNoYW5nZWRgKTtcbiAgICAgICAgY29uc29sZS5sb2coY2hhdE1lc3NhZ2VMaXN0KTtcbiAgICAgICAgdGhpcy5ub3RpZnkoXCJzb21lIHVzZXJcIik7XG4gICAgfVxuXG4gICAgcHVibGljIGFzeW5jIGdldG1lc3NhZ2VzKHVzZXJOYW1lOiBzdHJpbmcsIHBhc3NwaHJhc2U6IHN0cmluZywgbGFzdE1lc3NhZ2VUaW1lOiBzdHJpbmcgfCBudWxsKTogUHJvbWlzZTxDaGF0TWVzc2FnZVZpZXdNb2RlbFtdPiB7XG4gICAgICAgIGNvbnN0IGNWTXMgPSBhd2FpdCBDaGF0TW9kZWxIZWxwZXIuZ2V0TWVzc2FnZXModXNlck5hbWUsIHBhc3NwaHJhc2UsIGxhc3RNZXNzYWdlVGltZSwgdGhpcyk7XG4gICAgICAgIGlmIChjVk1zICE9IG51bGwpIHtcbiAgICAgICAgICAgIGxvZy5pbmZvKCdTdWJqZWN0OiBNeSBzdGF0ZSBoYXMganVzdCBjaGFuZ2VkJylcbiAgICAgICAgICAgIGxvZy5kZWJ1ZyhjVk1zKTtcbiAgICAgICAgICAgIHRoaXMuX21lc3NhZ2VzTWFwLnNldCh1c2VyTmFtZSwgY1ZNcyk7XG4gICAgICAgICAgICB0aGlzLm5vdGlmeSh1c2VyTmFtZSk7XG4gICAgICAgIH1cbiAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICBsb2cuZXJyb3IoJ01lc3NhZ2VzIHdlcmUgbnVsbCcpO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIGNWTXM7XG4gICAgfVxuXG5cblxuICAgIHByaXZhdGUgaGVscGVyTWV0aG9kKCkgeyB9XG5cbiAgICBwdWJsaWMgcG9wdWxhdGVNZXNzYWdlcygpOiB2b2lkIHtcblxuICAgIH1cblxuXG5cbn0iLCJpbXBvcnQgeyBDaGF0TWVzc2FnZURUTyB9IGZyb20gXCIuLi9kdG8vQ2hhdE1lc3NhZ2VEVE9cIjtcblxuaW1wb3J0ICogYXMgbG9nIGZyb20gXCJsb2dsZXZlbFwiO1xuXG5pbXBvcnQgeyBDaGF0TWVzc2FnZVZpZXdNb2RlbCB9IGZyb20gXCIuLi92aWV3bW9kZWwvQ2hhdE1lc3NhZ2VWaWV3TW9kZWxcIjtcblxuaW1wb3J0IHsgSnNvbkFQSSB9IGZyb20gXCIuLi9zaW5nbGV0b24vSnNvbkFQSVwiO1xuXG5pbXBvcnQgeyBmZXRjaEVycm9ySGFuZGxlciB9IGZyb20gXCIuL0ZldGNoRXJyb3JIYW5kbGVyXCI7XG5cbmltcG9ydCB7IEVuY3J5cHRpb25TZXJ2aWNlIH0gZnJvbSBcIi4uL3NlcnZpY2UvRW5jcnlwdGlvblNlcnZpY2VcIjtcbmltcG9ydCB7IFNKQ0xFbmNyeXB0aW9uU2VydmljZSB9IGZyb20gXCIuLi9zZXJ2aWNlL1NKQ0xFbmNyeXB0aW9uU2VydmljZVwiO1xuaW1wb3J0IHsgQ2hhdE1vZGVsIH0gZnJvbSBcIi4vQ2hhdE1vZGVsXCJcbmV4cG9ydCBjbGFzcyBDaGF0TW9kZWxIZWxwZXIge1xuICAgIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IGVuY3J5cHRpb25TZXJ2aWNlID0gbmV3IFNKQ0xFbmNyeXB0aW9uU2VydmljZSgpO1xuXG4gICAgcHVibGljIHN0YXRpYyBhc3luYyBnZXRNZXNzYWdlcyh1c2VyTmFtZTogc3RyaW5nLCBwYXNzcGhyYXNlOiBzdHJpbmcsIGxhc3RNZXNzYWdlVGltZTogc3RyaW5nIHwgbnVsbCwgY2hhdE1vZGVsOiBDaGF0TW9kZWwpOiBQcm9taXNlPENoYXRNZXNzYWdlVmlld01vZGVsW10+IHtcbiAgICAgICAgc3dpdGNoIChsYXN0TWVzc2FnZVRpbWUpIHtcbiAgICAgICAgICAgIGNhc2UgbnVsbDoge1xuICAgICAgICAgICAgICAgIC8vIHRoaXMuZ2V0QWxsTWVzc2FnZXNBamF4KHVzZXJOYW1lKVxuICAgICAgICAgICAgICAgIC8vICAgICAudGhlbigoZGF0YTogQ2hhdE1lc3NhZ2VEVE9bXSkgPT4ge1xuICAgICAgICAgICAgICAgIC8vICAgICAgICAgbG9nLmRlYnVnKGBTdWJqZWN0OiByZWNlaXZlZCBhbGwgbWVzc2FnZXNgKTtcbiAgICAgICAgICAgICAgICAvLyAgICAgICAgIC8vIGxldCB1c2VyTmFtZXMgPSBkYXRhLm1hcChDaGF0TWVzc2FnZVZpZXdNb2RlbCA9PiBDaGF0TWVzc2FnZVZpZXdNb2RlbC5mcm9tVXNlcilcbiAgICAgICAgICAgICAgICAvLyAgICAgICAgIC8vIGxldCBzdW10ID0gZGF0YS5tYXAoY2hhdE1lc3NhZ2VWaWV3TW9kZWwgPT4geyByZXR1cm4gdGhpcy5lbmNyeXB0aW9uU2VydmljZS5kZWNyeXB0KHBhc3NwaHJhc2UsIGNoYXRNZXNzYWdlVmlld01vZGVsLm1lc3NhZ2VDaXBoZXIpIH0pO1xuICAgICAgICAgICAgICAgIC8vICAgICAgICAgcmV0dXJuIGRhdGEubWFwKHZtID0+IHRoaXMudG9DaGF0TWVzc2FnZVZNKHZtLCBwYXNzcGhyYXNlKSk7XG4gICAgICAgICAgICAgICAgLy8gICAgICAgICAvLyBjaGF0TW9kZWwuc2V0VXNlck1lc3NhZ2VzKHVzZXJOYW1lLCBjaGF0TWVzc2FnZVZNcyk7XG4gICAgICAgICAgICAgICAgLy8gICAgICAgICAvLyBjaGF0TW9kZWwubm90aWZ5KCk7XG4gICAgICAgICAgICAgICAgLy8gICAgIH0pXG4gICAgICAgICAgICAgICAgLy8gYnJlYWs7XG5cbiAgICAgICAgICAgICAgICBjb25zdCBkYXRhOiBDaGF0TWVzc2FnZURUT1tdID0gYXdhaXQgdGhpcy5nZXRBbGxNZXNzYWdlc0FqYXgodXNlck5hbWUpO1xuICAgICAgICAgICAgICAgIHJldHVybiBkYXRhLm1hcCh2bSA9PiB0aGlzLnRvQ2hhdE1lc3NhZ2VWTSh2bSwgcGFzc3BocmFzZSkpO1xuXG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBkZWZhdWx0OiB7XG4gICAgICAgICAgICAgICAgLy8gdGhpcy5nZXROZXdNZXNzYWdlc0FqYXgodXNlck5hbWUsIGxhc3RNZXNzYWdlVGltZSlcbiAgICAgICAgICAgICAgICAvLyAgICAgLnRoZW4oKGRhdGE6IENoYXRNZXNzYWdlRFRPW10pID0+IHtcbiAgICAgICAgICAgICAgICAvLyAgICAgICAgIGxvZy5kZWJ1ZyhgU3ViamVjdDogcmVjZWl2ZWQgbmV3IG1lc3NhZ2VzYCk7XG4gICAgICAgICAgICAgICAgLy8gICAgICAgICByZXR1cm4gZGF0YS5tYXAodm0gPT4gdGhpcy50b0NoYXRNZXNzYWdlVk0odm0sIHBhc3NwaHJhc2UpKTtcbiAgICAgICAgICAgICAgICAvLyAgICAgICAgIC8vIGNoYXRNb2RlbC5zZXRVc2VyTWVzc2FnZXModXNlck5hbWUsIGNoYXRNZXNzYWdlVk1zKTtcbiAgICAgICAgICAgICAgICAvLyAgICAgICAgIC8vIHRoaXMuc3RhdGUgPSBkYXRhO1xuICAgICAgICAgICAgICAgIC8vICAgICAgICAgLy8gY2hhdE1vZGVsLm5vdGlmeSgpO1xuICAgICAgICAgICAgICAgIC8vICAgICB9KVxuICAgICAgICAgICAgICAgIC8vIGJyZWFrO1xuICAgICAgICAgICAgICAgIGNvbnN0IGRhdGE6IENoYXRNZXNzYWdlRFRPW10gPSBhd2FpdCB0aGlzLmdldE5ld01lc3NhZ2VzQWpheCh1c2VyTmFtZSwgbGFzdE1lc3NhZ2VUaW1lKTtcbiAgICAgICAgICAgICAgICByZXR1cm4gZGF0YS5tYXAodm0gPT4gdGhpcy50b0NoYXRNZXNzYWdlVk0odm0sIHBhc3NwaHJhc2UpKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIHByaXZhdGUgc3RhdGljIHRvQ2hhdE1lc3NhZ2VWTShjaGF0TWVzc2FnZURUTzogQ2hhdE1lc3NhZ2VEVE8sIHBhc3NwaHJhc2U6IHN0cmluZyk6IENoYXRNZXNzYWdlVmlld01vZGVsIHtcbiAgICAgICAgY29uc3Qgdm0gPSBuZXcgQ2hhdE1lc3NhZ2VWaWV3TW9kZWwoKTtcbiAgICAgICAgdm0uZnJvbVVzZXIgPSBjaGF0TWVzc2FnZURUTy5mcm9tVXNlcjtcbiAgICAgICAgdm0udG9Vc2VyID0gY2hhdE1lc3NhZ2VEVE8udG9Vc2VyO1xuICAgICAgICB2bS5tZXNzYWdlVGltZSA9IGNoYXRNZXNzYWdlRFRPLm1lc3NhZ2VUaW1lO1xuICAgICAgICB2bS5tZXNzYWdlID0gdGhpcy5lbmNyeXB0aW9uU2VydmljZS5kZWNyeXB0KHBhc3NwaHJhc2UsIGNoYXRNZXNzYWdlRFRPLm1lc3NhZ2VDaXBoZXIpIGFzIHN0cmluZztcbiAgICAgICAgcmV0dXJuIHZtO1xuICAgIH1cblxuICAgIHByaXZhdGUgc3RhdGljIGFzeW5jIGdldEFsbE1lc3NhZ2VzQWpheCh0b1VzZXI6IHN0cmluZyk6IFByb21pc2U8YW55PiB7XG4gICAgICAgIGNvbnN0IGhlYWRlcnMgPSBuZXcgSGVhZGVycygpO1xuICAgICAgICBpZiAoSnNvbkFQSS5hdXRoVG9rZW4gPT0gbnVsbCkge1xuICAgICAgICAgICAgbG9nLmVycm9yKFwiYXV0aFRva2VuIG51bGxcIik7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH07XG4gICAgICAgIGhlYWRlcnMuYXBwZW5kKCdYLUFVVEgtVE9LRU4nLCBKc29uQVBJLmF1dGhUb2tlbik7XG4gICAgICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2goYCR7SnNvbkFQSS5DSEFUX01FU1NBR0VTX0dFVH0vJHt0b1VzZXJ9YCwge1xuICAgICAgICAgICAgbWV0aG9kOiAnR0VUJyxcbiAgICAgICAgICAgIGhlYWRlcnM6IGhlYWRlcnNcbiAgICAgICAgfSk7XG4gICAgICAgIGNvbnNvbGUubG9nKHJlc3BvbnNlLmNsb25lKCkpO1xuICAgICAgICBpZiAoZmV0Y2hFcnJvckhhbmRsZXIocmVzcG9uc2UuY2xvbmUoKSkpIHtcbiAgICAgICAgICAgIHJldHVybiBudWxsO1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IGRhdGE6IFByb21pc2U8YW55PiA9IGF3YWl0IHJlc3BvbnNlLmpzb24oKTtcbiAgICAgICAgcmV0dXJuIGRhdGE7XG4gICAgfVxuXG4gICAgcHJpdmF0ZSBzdGF0aWMgYXN5bmMgZ2V0TmV3TWVzc2FnZXNBamF4KHRvVXNlcjogc3RyaW5nLCBsYXN0TWVzc2FnZVRpbWVTdGFtcDogc3RyaW5nKTogUHJvbWlzZTxhbnk+IHtcbiAgICAgICAgY29uc3QgaGVhZGVycyA9IG5ldyBIZWFkZXJzKCk7XG4gICAgICAgIGlmIChKc29uQVBJLmF1dGhUb2tlbiA9PSBudWxsKSB7XG4gICAgICAgICAgICBsb2cuZXJyb3IoXCJhdXRoVG9rZW4gbnVsbFwiKTtcbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfTtcbiAgICAgICAgaGVhZGVycy5hcHBlbmQoJ1gtQVVUSC1UT0tFTicsIEpzb25BUEkuYXV0aFRva2VuKTtcbiAgICAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBmZXRjaChgJHtKc29uQVBJLkNIQVRfTUVTU0FHRVNfR0VUfS8ke3RvVXNlcn0vJHtsYXN0TWVzc2FnZVRpbWVTdGFtcH1gLCB7XG4gICAgICAgICAgICBtZXRob2Q6ICdHRVQnLFxuICAgICAgICAgICAgaGVhZGVyczogaGVhZGVyc1xuICAgICAgICB9KTtcbiAgICAgICAgY29uc29sZS5sb2cocmVzcG9uc2UuY2xvbmUoKSk7XG4gICAgICAgIGlmIChmZXRjaEVycm9ySGFuZGxlcihyZXNwb25zZS5jbG9uZSgpKSkge1xuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cbiAgICAgICAgY29uc3QgZGF0YTogUHJvbWlzZTxhbnk+ID0gYXdhaXQgcmVzcG9uc2UuanNvbigpO1xuICAgICAgICByZXR1cm4gZGF0YTtcbiAgICB9XG59IiwiaW1wb3J0IGxvZyA9IHJlcXVpcmUoXCJsb2dsZXZlbFwiKTtcbmltcG9ydCB7IHNwcmludGYgfSBmcm9tIFwic3ByaW50Zi1qc1wiO1xuLy8gaW1wb3J0IHNwcmludGYgPSByZXF1aXJlKCdzcHJpbnRmLWpzJykuc3ByaW50ZjtcblxuZXhwb3J0IGZ1bmN0aW9uIGZldGNoRXJyb3JIYW5kbGVyKHJlc3BvbnNlOiBSZXNwb25zZSkge1xuICAgIC8vIGFsZXJ0aWZ5LnN1Y2Nlc3MoJ0N1cnJlbnQgcG9zaXRpb24gOiAnICsgYWxlcnRpZnkuZ2V0KCdub3RpZmllcicsICdwb3NpdGlvbicpKTtcbiAgICBpZiAoIXJlc3BvbnNlLm9rKSB7XG4gICAgICAgIHJldHVybiByZXNwb25zZS50ZXh0KCkuY2F0Y2goZXJyID0+IHtcbiAgICAgICAgICAgIC8vIHRoZSBzdGF0dXMgd2FzIG5vdCBvayBhbmQgdGhlcmUgaXMgbm8ganNvbiBib2R5XG4gICAgICAgICAgICAvLyB0aHJvdyBuZXcgRXJyb3IocmVzcG9uc2Uuc3RhdHVzVGV4dCk7XG4gICAgICAgICAgICAvLyB3aW5kb3cuYWxlcnQoc3ByaW50ZignU29tZSBlcnJvciBvY2N1cmVkLiBIdHRwIGNvZGUgaXMgJXMnLCByZXNwb25zZS5zdGF0dXMpKTtcbiAgICAgICAgICAgIC8vIGFsZXJ0aWZ5LmVycm9yKHNwcmludGYoJ1NvbWUgZXJyb3Igb2NjdXJlZC4gSHR0cCBjb2RlIGlzICVzJywgcmVzcG9uc2Uuc3RhdHVzKSk7XG4gICAgICAgICAgICBsb2cuZXJyb3Ioc3ByaW50ZignU29tZSBlcnJvciBvY2N1cmVkLiBIdHRwIGNvZGUgaXMgJXMnLCByZXNwb25zZS5zdGF0dXMpKTtcbiAgICAgICAgICAgIGxvZy5lcnJvcigpO1xuICAgICAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgICAgIH0pLnRoZW4oanNvbiA9PiB7XG4gICAgICAgICAgICAvLyB0aGUgc3RhdHVzIHdhcyBub3Qgb2sgYnV0IHRoZXJlIGlzIGEganNvbiBib2R5XG4gICAgICAgICAgICAvLyB0aHJvdyBuZXcgRXJyb3IoanNvbi5lcnJvci5tZXNzYWdlKTsgLy8gZXhhbXBsZSBlcnJvciBtZXNzYWdlIHJldHVybmVkIGJ5IGEgUkVTVCBBUElcbiAgICAgICAgICAgIC8vIHdpbmRvdy5hbGVydChzcHJpbnRmKCdFcnJvcjogJXMgKEh0dHAgY29kZSAlcyknLCBqc29uLCByZXNwb25zZS5zdGF0dXMpKTtcbiAgICAgICAgICAgIC8vIGFsZXJ0aWZ5LmVycm9yKHNwcmludGYoJ1NvbWUgZXJyb3Igb2NjdXJlZC4gSHR0cCBjb2RlIGlzICVzJywgcmVzcG9uc2Uuc3RhdHVzKSk7XG4gICAgICAgICAgICBsb2cuZXJyb3Ioc3ByaW50ZignU29tZSBlcnJvciBvY2N1cmVkLiBIdHRwIGNvZGUgaXMgJXMnLCByZXNwb25zZS5zdGF0dXMpKTtcbiAgICAgICAgICAgIGxvZy5lcnJvcihqc29uKTtcbiAgICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICB9KTtcbiAgICB9XG59IiwiaW1wb3J0IHsgU3ViamVjdCB9IGZyb20gXCIuLi9vYnNlcnZlL09ic2VydmFibGVcIjtcbmltcG9ydCB7IE1vZGVsIH0gZnJvbSBcIi4vQWJzdHJhY3RNb2RlbFwiO1xuaW1wb3J0IHsgT2JzZXJ2ZXIgfSBmcm9tIFwiLi4vb2JzZXJ2ZS9PYnNlcnZlclwiO1xuaW1wb3J0IHsgZmV0Y2hFcnJvckhhbmRsZXIgfSBmcm9tIFwiLi9GZXRjaEVycm9ySGFuZGxlclwiO1xuaW1wb3J0IHsgQWN0aXZlVXNlclZpZXdNb2RlbCB9IGZyb20gXCIuLi92aWV3bW9kZWwvQWN0aXZlVXNlclZpZXdNb2RlbFwiO1xuaW1wb3J0IHsgSnNvbkFQSSB9IGZyb20gXCIuLi9zaW5nbGV0b24vSnNvbkFQSVwiO1xuaW1wb3J0IHsgQ2hhdE1lc3NhZ2VWaWV3TW9kZWwgfSBmcm9tIFwiLi4vdmlld21vZGVsL0NoYXRNZXNzYWdlVmlld01vZGVsXCI7XG5pbXBvcnQgKiBhcyBsb2cgZnJvbSBcImxvZ2xldmVsXCI7XG5cbmV4cG9ydCBjbGFzcyBVc2VyTW9kZWwgaW1wbGVtZW50cyBTdWJqZWN0IHtcbiAgICAvKipcbiAgKiBAdHlwZSB7T2JzZXJ2ZXJbXX0gTGlzdCBvZiBzdWJzY3JpYmVycy4gSW4gcmVhbCBsaWZlLCB0aGUgbGlzdCBvZlxuICAqIHN1YnNjcmliZXJzIGNhbiBiZSBzdG9yZWQgbW9yZSBjb21wcmVoZW5zaXZlbHkgKGNhdGVnb3JpemVkIGJ5IGV2ZW50XG4gICogdHlwZSwgZXRjLikuXG4gICovXG4gICAgcHJpdmF0ZSByZWFkb25seSBvYnNlcnZlcnM6IE9ic2VydmVyW10gPSBbXTtcbiAgICBwcml2YXRlIHN0YXRlOiBBY3RpdmVVc2VyVmlld01vZGVsW10gfCB1bmRlZmluZWQ7XG4gICAgLy8gQHRzLWlnbm9yZTogQ2Fubm90IGZpbmQgbmFtZSAnaG9zdEFkZHJlc3MnLlxuXG4gICAgY29uc3RydWN0b3IoKSB7IH1cbiAgICAvKipcbiAgICAgKiBUaGUgc3Vic2NyaXB0aW9uIG1hbmFnZW1lbnQgbWV0aG9kcy5cbiAgICAgKi9cbiAgICBwdWJsaWMgYXR0YWNoKG9ic2VydmVyOiBPYnNlcnZlcik6IHZvaWQge1xuICAgICAgICBjb25zb2xlLmxvZygnU3ViamVjdDogQXR0YWNoZWQgYW4gb2JzZXJ2ZXIuJyk7XG4gICAgICAgIHRoaXMub2JzZXJ2ZXJzLnB1c2gob2JzZXJ2ZXIpO1xuICAgIH1cblxuICAgIHB1YmxpYyBkZXRhY2gob2JzZXJ2ZXI6IE9ic2VydmVyKTogdm9pZCB7XG4gICAgICAgIGNvbnN0IG9ic2VydmVySW5kZXggPSB0aGlzLm9ic2VydmVycy5pbmRleE9mKG9ic2VydmVyKTtcbiAgICAgICAgdGhpcy5vYnNlcnZlcnMuc3BsaWNlKG9ic2VydmVySW5kZXgsIDEpO1xuICAgICAgICBjb25zb2xlLmxvZygnU3ViamVjdDogRGV0YWNoZWQgYW4gb2JzZXJ2ZXIuJyk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogVHJpZ2dlciBhbiB1cGRhdGUgaW4gZWFjaCBzdWJzY3JpYmVyLlxuICAgICAqL1xuICAgIHB1YmxpYyBub3RpZnkoKTogdm9pZCB7XG4gICAgICAgIGNvbnNvbGUubG9nKCdTdWJqZWN0OiBOb3RpZnlpbmcgb2JzZXJ2ZXJzLi4uJyk7XG4gICAgICAgIGZvciAoY29uc3Qgb2JzZXJ2ZXIgb2YgdGhpcy5vYnNlcnZlcnMpIHtcbiAgICAgICAgICAgIG9ic2VydmVyLnVwZGF0ZSh0aGlzLnN0YXRlKTtcbiAgICAgICAgfVxuICAgIH1cblxuICAgIHB1YmxpYyBzb21lQnVzaW5lc3NNZXRob2QoYWN0aXZldXNlckxpc3Q6IEFjdGl2ZVVzZXJWaWV3TW9kZWxbXSk6IHZvaWQge1xuICAgICAgICB0aGlzLnN0YXRlID0gYWN0aXZldXNlckxpc3Q7XG4gICAgICAgIHRoaXMuaGVscGVyTWV0aG9kKCk7XG4gICAgICAgIGNvbnNvbGUubG9nKGBTdWJqZWN0OiBNeSBzdGF0ZSBoYXMganVzdCBjaGFuZ2VkYCk7XG4gICAgICAgIGNvbnNvbGUubG9nKGFjdGl2ZXVzZXJMaXN0KTtcbiAgICAgICAgdGhpcy5ub3RpZnkoKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBnZXRBY3RpdmVVc2Vyc1xuICAgICAqL1xuICAgIHB1YmxpYyBnZXRBY3RpdmVVc2VycygpOiB2b2lkIHtcbiAgICAgICAgaWYoSnNvbkFQSS5hdXRoVG9rZW4hPSBudWxsKXtcbiAgICAgICAgdGhpcy5nZXRBY3RpdmVVc2Vyc0FqYXgoSnNvbkFQSS5hdXRoVG9rZW4sIEpzb25BUEkuQUNUSVZFX1VTRVJTX0dFVClcbiAgICAgICAgICAgIC50aGVuKGRhdGEgPT4ge1xuICAgICAgICAgICAgICAgIC8vIC8vIGFjdGl2ZVVzZXJzID0gZGF0YTtcbiAgICAgICAgICAgICAgICAvLyBzZXNzaW9uU3RvcmFnZS5zZXRJdGVtKCdhY3RpdmVVc2VycycsIEpTT04uc3RyaW5naWZ5KGRhdGEpKTtcbiAgICAgICAgICAgICAgICAvLyBjb25zb2xlLmxvZyhzZXNzaW9uU3RvcmFnZS5nZXRJdGVtKCdhY3RpdmVVc2VycycpKTtcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhgU3ViamVjdDogcmVjZWl2ZWQgYWpheCBhY3RpdmUgdXNlcnNgKTtcbiAgICAgICAgICAgICAgICB0aGlzLnN0YXRlID0gZGF0YTtcbiAgICAgICAgICAgICAgICB0aGlzLm5vdGlmeSgpO1xuICAgICAgICAgICAgfSlcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGxvZy5lcnJvcignQXV0aCB0b2tlbiBpcyBudWxsJyk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBhc3luYyBnZXRBY3RpdmVVc2Vyc0FqYXgoYXV0aFRva2VuMjogc3RyaW5nLCBVUkw6IHN0cmluZyk6IFByb21pc2U8YW55PiB7XG4gICAgICAgIGxldCBoZWFkZXJzID0gbmV3IEhlYWRlcnMoKTtcbiAgICAgICAgLy8gaGVhZGVycy5hcHBlbmQoJ0F1dGhvcml6YXRpb24nLCBiYXNpY0F1dGhUb2tlbik7XG4gICAgICAgIGhlYWRlcnMuYXBwZW5kKCdYLUFVVEgtVE9LRU4nLCBhdXRoVG9rZW4yKTtcbiAgICAgICAgbGV0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2goSnNvbkFQSS5BQ1RJVkVfVVNFUlNfR0VULCB7XG4gICAgICAgICAgICBtZXRob2Q6ICdHRVQnLFxuICAgICAgICAgICAgaGVhZGVyczogaGVhZGVyc1xuICAgICAgICB9KTtcbiAgICAgICAgY29uc29sZS5sb2cocmVzcG9uc2UuY2xvbmUoKSk7XG4gICAgICAgIGlmIChmZXRjaEVycm9ySGFuZGxlcihyZXNwb25zZS5jbG9uZSgpKSkge1xuICAgICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cbiAgICAgICAgbGV0IGRhdGEgPSBhd2FpdCByZXNwb25zZS5qc29uKCk7XG4gICAgICAgIC8vIHJldHVybiBkYXRhO1xuICAgICAgICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgICAgICAgICAgaWYgKGRhdGEgIT0gbnVsbClcbiAgICAgICAgICAgICAgICByZXNvbHZlKGRhdGEpXG4gICAgICAgICAgICBlbHNlXG4gICAgICAgICAgICAgICAgcmVqZWN0KCdSZXNwb25zZSBkYXRhIG51bGwnKVxuICAgICAgICB9KVxuICAgIH1cblxuICAgIHByaXZhdGUgaGVscGVyTWV0aG9kKCkgeyB9XG5cbn0iLCJpbXBvcnQgeyBFbmNyeXB0aW9uU2VydmljZSB9IGZyb20gXCIuL0VuY3J5cHRpb25TZXJ2aWNlXCI7XG5pbXBvcnQgKiBhcyBzamNsIGZyb20gXCJzamNsXCI7XG5cbmV4cG9ydCBjbGFzcyBTSkNMRW5jcnlwdGlvblNlcnZpY2UgaW1wbGVtZW50cyBFbmNyeXB0aW9uU2VydmljZSB7XG4gICAgcHJpdmF0ZSByZWFkb25seSBwYXJhbXM6IGFueSA9IHsgbW9kZTogXCJnY21cIiwgdHM6IDEyOCwgYWRhdGE6IFwiXCIsIGl0ZXI6IDEwMDAwfVxuICAgIHB1YmxpYyBlbmNyeXB0KHBhc3NwaHJhc2U6IHN0cmluZywgcGxhaW5UZXh0OiBzdHJpbmcpOiBPYmplY3Qge1xuICAgICAgICByZXR1cm4gc2pjbC5lbmNyeXB0KHBhc3NwaHJhc2UsIHBsYWluVGV4dCwgdGhpcy5wYXJhbXMpO1xuICAgIH0gXG4gICAgXG4gICAgcHVibGljIGRlY3J5cHQocGFzc3BocmFzZTogc3RyaW5nLCBjaXBoZXI6IE9iamVjdCk6IE9iamVjdCB7XG4gICAgICAgIC8vIHJldHVybiBzamNsLmRlY3J5cHQocGFzc3BocmFzZSwgY2lwaGVyIGFzIHNqY2wuU2pjbENpcGhlckVuY3J5cHRlZCwgdW5kZWZpbmVkLCB1bmRlZmluZWQpO1xuICAgICAgICByZXR1cm4gc2pjbC5kZWNyeXB0KHBhc3NwaHJhc2UsIEpTT04uc3RyaW5naWZ5KGNpcGhlciksIHVuZGVmaW5lZCwgdW5kZWZpbmVkKTtcbiAgICB9XG59IiwiZXhwb3J0IG5hbWVzcGFjZSBKc29uQVBJIHtcbiAgICAvLyBAdHMtaWdub3JlOiBDYW5ub3QgZmluZCBuYW1lICdob3N0QWRkcmVzcycuXG4gICAgZXhwb3J0IGxldCB1c2VyTmFtZTogc3RyaW5nIHwgbnVsbCA9IGxvY2FsU3RvcmFnZS5nZXRJdGVtKCd1c2VyTmFtZScpO1xuICAgIGV4cG9ydCBsZXQgYXV0aFRva2VuOiBzdHJpbmcgfCBudWxsID0gbG9jYWxTdG9yYWdlLmdldEl0ZW0oJ2F1dGhUb2tlbicpO1xuICAgIGV4cG9ydCBjb25zdCBBQ1RJVkVfVVNFUlNfR0VUID0gYC9hcGkvY2hhdC9nZXQvYWN0aXZlLXVzZXJzYDtcbiAgICBleHBvcnQgY29uc3QgQ0hBVF9NRVNTQUdFU19HRVQgPSBgL2FwaS9jaGF0L2dldC9tZXNzYWdlc2A7XG4gICAgXG59IiwiZXhwb3J0IGNsYXNzIFRlbXBsYXRlRmFjdG9yeSB7XG4gICAgLy8gc3RhdGljIGdldFRlbXBsYXRlKHRlbXBsYXRlTmFtZTogc3RyaW5nKTogSGFuZGxlYmFycy5UZW1wbGF0ZURlbGVnYXRlPGFueT4ge1xuXG4gICAgLy8gICAgIHN3aXRjaCAodGVtcGxhdGVOYW1lKSB7XG4gICAgLy8gICAgICAgICBjYXNlIFwidXNlci1jb250YWN0LW9ubGluZS10ZW1wbGF0ZVwiOiB7XG4gICAgLy8gICAgICAgICAgICAgLy8gbGV0IHNvdXJjZSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKFwidXNlci1jb250YWN0LW9ubGluZS10ZW1wbGF0ZVwiKS5pbm5lckhUTUw7XG4gICAgLy8gICAgICAgICAgICAgLy8gbGV0IG1zZ0NvbnRhaW5lclRlbXBsYXRlID0gSGFuZGxlYmFycy5jb21waWxlKHNvdXJjZSk7XG4gICAgLy8gICAgICAgICAgICAgLy8gcmV0dXJuIG1zZ0NvbnRhaW5lclRlbXBsYXRlO1xuICAgIC8vICAgICAgICAgICAgIHJldHVybiB0aGlzLmNyZWF0ZVRlbXBsYXRlKHRlbXBsYXRlTmFtZSk7XG4gICAgLy8gICAgICAgICB9XG4gICAgLy8gICAgICAgICBjYXNlIFwibXNnX2NvbnRhaW5lcl9zZW5kX3RlbXBsYXRlXCI6IHtcbiAgICAvLyAgICAgICAgICAgICAvLyBsZXQgc291cmNlID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoXCJtc2dfY29udGFpbmVyX3NlbmRfdGVtcGxhdGVcIikuaW5uZXJIVE1MO1xuICAgIC8vICAgICAgICAgICAgIC8vIGxldCBtc2dDb250YWluZXJUZW1wbGF0ZSA9IEhhbmRsZWJhcnMuY29tcGlsZShzb3VyY2UpO1xuICAgIC8vICAgICAgICAgICAgIC8vIHJldHVybiBtc2dDb250YWluZXJUZW1wbGF0ZTtcbiAgICAvLyAgICAgICAgICAgICByZXR1cm4gdGhpcy5jcmVhdGVUZW1wbGF0ZSh0ZW1wbGF0ZU5hbWUpO1xuICAgIC8vICAgICAgICAgfVxuICAgIC8vICAgICAgICAgY2FzZSBcIm1zZ19jb250YWluZXJfdGVtcGxhdGVcIjoge1xuICAgIC8vICAgICAgICAgICAgIC8vIGxldCBzb3VyY2UgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChcIm1zZ19jb250YWluZXJfc2VuZF90ZW1wbGF0ZVwiKS5pbm5lckhUTUw7XG4gICAgLy8gICAgICAgICAgICAgLy8gbGV0IG1zZ0NvbnRhaW5lclRlbXBsYXRlID0gSGFuZGxlYmFycy5jb21waWxlKHNvdXJjZSk7XG4gICAgLy8gICAgICAgICAgICAgLy8gcmV0dXJuIG1zZ0NvbnRhaW5lclRlbXBsYXRlO1xuICAgIC8vICAgICAgICAgICAgIHJldHVybiB0aGlzLmNyZWF0ZVRlbXBsYXRlKHRlbXBsYXRlTmFtZSk7XG4gICAgLy8gICAgICAgICB9XG4gICAgLy8gICAgICAgICBkZWZhdWx0OlxuICAgIC8vICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcignaW52YWxpZCB0ZW1wbGF0ZSBuYW1lJyk7XG4gICAgLy8gICAgIH1cbiAgICAvLyB9XG5cbiAgICBzdGF0aWMgZ2V0VGVtcGxhdGUodGVtcGxhdGVOYW1lOiBzdHJpbmcpOiBIYW5kbGViYXJzLlRlbXBsYXRlRGVsZWdhdGU8YW55PiB7XG4gICAgICAgIC8vQHRzLWlnbm9yZTogT2JqZWN0IGlzIHBvc3NpYmx5ICdudWxsJy5cbiAgICAgICAgbGV0IHNvdXJjZSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKHRlbXBsYXRlTmFtZSkuaW5uZXJIVE1MO1xuICAgICAgICBsZXQgbXNnQ29udGFpbmVyVGVtcGxhdGUgPSBIYW5kbGViYXJzLmNvbXBpbGUoc291cmNlKTtcbiAgICAgICAgcmV0dXJuIG1zZ0NvbnRhaW5lclRlbXBsYXRlO1xuICAgIH1cbn0iLCJpbXBvcnQgeyBPYnNlcnZlciB9IGZyb20gXCIuLi9vYnNlcnZlL09ic2VydmVyXCI7XG5pbXBvcnQgeyBNb2RlbCB9IGZyb20gXCIuLi9tb2RlbC9BYnN0cmFjdE1vZGVsXCI7XG5pbXBvcnQgeyBTdWJqZWN0IH0gZnJvbSBcIi4uL29ic2VydmUvT2JzZXJ2YWJsZVwiO1xuaW1wb3J0IHsgVmlldyB9IGZyb20gXCIuL0Fic3RyYWN0Vmlld1wiO1xuaW1wb3J0IHsgQ29udHJvbGxlciB9IGZyb20gXCIuLi9jb250cm9sbGVyL0Fic3RyYWN0Q29udHJvbGxlclwiO1xuaW1wb3J0IHsgVGVtcGxhdGVGYWN0b3J5IH0gZnJvbSBcIi4uL3RlbXBsYXRlL1RlbXBsYXRlRmFjdG9yeVwiO1xuaW1wb3J0IHsgQWN0aXZlVXNlclZpZXdNb2RlbCB9IGZyb20gXCIuLi92aWV3bW9kZWwvQWN0aXZlVXNlclZpZXdNb2RlbFwiO1xuaW1wb3J0IHsgQ2hhdE1vZGVsIH0gZnJvbSBcIi4uL21vZGVsL0NoYXRNb2RlbFwiO1xuaW1wb3J0IHsgQ2hhdE1lc3NhZ2VWaWV3TW9kZWwgfSBmcm9tIFwiLi4vdmlld21vZGVsL0NoYXRNZXNzYWdlVmlld01vZGVsXCI7XG5pbXBvcnQgKiBhcyBsb2cgZnJvbSAnbG9nbGV2ZWwnO1xuaW1wb3J0ICogYXMgRE9NUHVyaWZ5IGZyb20gJ2RvbXB1cmlmeSc7XG5pbXBvcnQgbWFya2Rvd25pdCA9IHJlcXVpcmUoJ21hcmtkb3duLWl0Jyk7XG52YXIgbWQgPSBuZXcgbWFya2Rvd25pdCgpO1xuXG5leHBvcnQgY2xhc3MgQ2hhdFZpZXcgaW1wbGVtZW50cyBPYnNlcnZlciB7XG4gICAgcHJpdmF0ZSByZWFkb25seSBfbW9kZWw6IENoYXRNb2RlbDtcbiAgICBwcml2YXRlIHJlYWRvbmx5IF9lbGVtZW50OiBIVE1MRWxlbWVudDtcbiAgICBwcml2YXRlIHJlYWRvbmx5IF9tZXNzYWdlU2VuZFRlbXBsYXRlID0gVGVtcGxhdGVGYWN0b3J5LmdldFRlbXBsYXRlKCdtc2dfY29udGFpbmVyX3NlbmRfdGVtcGxhdGUnKTtcblxuXG4gICAgY29uc3RydWN0b3IobW9kZWw6IENoYXRNb2RlbCwgZWxlbWVudDogSFRNTEVsZW1lbnQpIHtcbiAgICAgICAgdGhpcy5fbW9kZWwgPSBtb2RlbDtcbiAgICAgICAgdGhpcy5fZWxlbWVudCA9IGVsZW1lbnQ7XG4gICAgfVxuXG5cbiAgICB1cGRhdGUoZGF0YTogQ2hhdE1lc3NhZ2VWaWV3TW9kZWxbXSk6IHZvaWQge1xuICAgICAgICBsb2cuaW5mbygnQ2hhdFZpZXc6IHVwZGF0aW5nIHZpZXcnKTtcbiAgICAgICAgbGV0IGh0bWw6IHN0cmluZyA9IFwiXCI7XG4gICAgICAgIGRhdGEuZm9yRWFjaCgodm06IENoYXRNZXNzYWdlVmlld01vZGVsKSA9PiB7XG4gICAgICAgICAgICBodG1sICs9IHRoaXMuX21lc3NhZ2VTZW5kVGVtcGxhdGUodm0pO1xuICAgICAgICB9KTtcblxuICAgICAgICAvKiogVmVyeSBJbXBvcnRhbnQhISFcbiAgICAgICAgICogU2FuaXRpemluZyBIVE1MIGJlZm9yZSBkaXNwbGF5aW5nIG9uIHdlYnBhZ2UgdG8gcHJldmVudCBYU1MgYXR0YWNrcyEhXG4gICAgICAgICAqL1xuICAgICAgICBodG1sID0gRE9NUHVyaWZ5LnNhbml0aXplKG1kLnJlbmRlcihodG1sKSk7XG4gICAgICAgIHRoaXMuX2VsZW1lbnQuaW5uZXJIVE1MID0gaHRtbDtcbiAgICAgICAgbG9nLmRlYnVnKHRoaXMuX2VsZW1lbnQuaW5uZXJIVE1MKTtcbiAgICB9XG59IiwiaW1wb3J0IHsgT2JzZXJ2ZXIgfSBmcm9tIFwiLi4vb2JzZXJ2ZS9PYnNlcnZlclwiO1xuaW1wb3J0IHsgTW9kZWwgfSBmcm9tIFwiLi4vbW9kZWwvQWJzdHJhY3RNb2RlbFwiO1xuaW1wb3J0IHsgU3ViamVjdCB9IGZyb20gXCIuLi9vYnNlcnZlL09ic2VydmFibGVcIjtcbmltcG9ydCB7IFZpZXcgfSBmcm9tIFwiLi9BYnN0cmFjdFZpZXdcIjtcbmltcG9ydCB7IENvbnRyb2xsZXIgfSBmcm9tIFwiLi4vY29udHJvbGxlci9BYnN0cmFjdENvbnRyb2xsZXJcIjtcbmltcG9ydCB7IFRlbXBsYXRlRmFjdG9yeSB9IGZyb20gXCIuLi90ZW1wbGF0ZS9UZW1wbGF0ZUZhY3RvcnlcIjtcbmltcG9ydCB7IEFjdGl2ZVVzZXJWaWV3TW9kZWwgfSBmcm9tIFwiLi4vdmlld21vZGVsL0FjdGl2ZVVzZXJWaWV3TW9kZWxcIjtcbmltcG9ydCB7IENoYXRNb2RlbCB9IGZyb20gXCIuLi9tb2RlbC9DaGF0TW9kZWxcIjtcbmltcG9ydCBsb2cgPSByZXF1aXJlKFwibG9nbGV2ZWxcIik7XG5cbmV4cG9ydCBjbGFzcyBVc2VyVmlldyBpbXBsZW1lbnRzIE9ic2VydmVyIHtcbiAgICBwcml2YXRlIHJlYWRvbmx5IF9tb2RlbDogTW9kZWw7XG4gICAgcHJpdmF0ZSByZWFkb25seSBfY2hhdE1vZGVsOiBDaGF0TW9kZWw7XG4gICAgcHJpdmF0ZSByZWFkb25seSBfZWxlbWVudDogSFRNTEVsZW1lbnQ7XG4gICAgLy8gcHJpdmF0ZSB1c2VyQm94ZXM6IGFueVtdID0gIFtdO1xuXG5cbiAgICBjb25zdHJ1Y3Rvcihtb2RlbDogTW9kZWwsIGNoYXRNb2RlbDogQ2hhdE1vZGVsLCBlbGVtZW50OiBIVE1MRWxlbWVudCkge1xuICAgICAgICB0aGlzLl9tb2RlbCA9IG1vZGVsO1xuICAgICAgICB0aGlzLl9jaGF0TW9kZWwgPSBjaGF0TW9kZWw7XG4gICAgICAgIHRoaXMuX2VsZW1lbnQgPSBlbGVtZW50O1xuICAgIH1cblxuXG5cblxuICAgIHVwZGF0ZShkYXRhOiBBY3RpdmVVc2VyVmlld01vZGVsW10pOiB2b2lkIHtcbiAgICAgICAgbGV0IHRlbXBsYXRlID0gVGVtcGxhdGVGYWN0b3J5LmdldFRlbXBsYXRlKCd1c2VyLWNvbnRhY3Qtb25saW5lLXRlbXBsYXRlJyk7XG4gICAgICAgIGxldCBodG1sOiBzdHJpbmcgPSBcIlwiO1xuICAgICAgICBkYXRhLmZvckVhY2goKGVsZW1lbnQ6IEFjdGl2ZVVzZXJWaWV3TW9kZWwpID0+IHtcbiAgICAgICAgICAgIGh0bWwgKz0gdGVtcGxhdGUoZWxlbWVudCk7XG4gICAgICAgIH0pO1xuICAgICAgICB0aGlzLl9lbGVtZW50LmlubmVySFRNTCA9IGh0bWw7XG4gICAgICAgIHRoaXMuYWRkVXNlckNhbGxCYWNrcygpO1xuICAgICAgICBjb25zb2xlLmxvZyh0aGlzLl9lbGVtZW50LmlubmVySFRNTCk7XG4gICAgfVxuXG4gICAgcHJpdmF0ZSBoZWxwZXIoKTogdm9pZCB7XG5cbiAgICB9XG5cbiAgICBwcml2YXRlIGFkZFVzZXJDYWxsQmFja3MoKTogdm9pZCB7XG4gICAgICAgIGxldCB1c2VyQm94ZXMgPSBkb2N1bWVudC5nZXRFbGVtZW50c0J5Q2xhc3NOYW1lKCd1c2VyLWJveCcpO1xuICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IHVzZXJCb3hlcy5sZW5ndGg7IGkrKykge1xuICAgICAgICAgICAgbGV0IHVzZXJCb3ggPSB1c2VyQm94ZXNbaV07XG4gICAgICAgICAgICB1c2VyQm94ZXNbaV0uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCB0aGlzLnVzZXJDYWxsQmFjay5iaW5kKHRoaXMsIHVzZXJCb3gpKTtcbiAgICAgICAgfVxuICAgIH1cblxuXG4gICAgcHJpdmF0ZSB1c2VyQ2FsbEJhY2soZWw6IEVsZW1lbnQpOiB2b2lkIHtcbiAgICAgICAgbGV0IGN1cnJlbnQgPSBkb2N1bWVudC5nZXRFbGVtZW50c0J5Q2xhc3NOYW1lKCd1c2VyLWJveCBhY3RpdmUnKTtcblxuICAgICAgICBsZXQgcGFzc3BocmFzZTogc3RyaW5nID0gJyc7XG4gICAgICAgIGlmIChjdXJyZW50Lmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgIGxldCBwYXNzcGhyYXNlSW5wdXQgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgncGFzc3BocmFzZScpIGFzIGFueTtcblxuICAgICAgICAgICAgaWYgKHBhc3NwaHJhc2VJbnB1dCA9PSBudWxsKSB7XG4gICAgICAgICAgICAgICAgbG9nLmVycm9yKCdwYXNzcGhyYXNlSW5wdXQgZWxlbWVudCByZWZlcmVuY2UgaXMgbnVsbCcpO1xuICAgICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHBhc3NwaHJhc2UgPSBwYXNzcGhyYXNlSW5wdXQudmFsdWVcbiAgICAgICAgICAgIGlmIChwYXNzcGhyYXNlID09ICcnIHx8IHBhc3NwaHJhc2UgPT0gbnVsbCkge1xuICAgICAgICAgICAgICAgIC8vIGFsZXJ0KCdQbGVhc2UgaW5wdXQgcGFzc3BocmFzZScpXG4gICAgICAgICAgICAgICAgLy8gYWxlcnRpZnkuZXJyb3IoJ1BsZWFzZSBlbnRlciBhIHBhc3NwaHJhc2UnKTtcbiAgICAgICAgICAgICAgICBsb2cuZXJyb3IoJ3Bhc3NwaHJhc2UgaXMgZW1wdHkgb3IgbnVsbCcpO1xuICAgICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGN1cnJlbnRbMF0uY2xhc3NOYW1lID0gY3VycmVudFswXS5jbGFzc05hbWUucmVwbGFjZShcIiBhY3RpdmVcIiwgXCJcIik7XG5cbiAgICAgICAgfVxuICAgICAgICAvLyBBZGQgdGhlIGFjdGl2ZSBjbGFzcyB0byB0aGUgY3VycmVudC9jbGlja2VkIGJ1dHRvblxuICAgICAgICBlbHNlIGlmIChjdXJyZW50Lmxlbmd0aCA9PSAwKSB7XG4gICAgICAgICAgICBsZXQgZWxlbSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdwYXNzcGhyYXNlLWluaXRpYWwnKSBhcyBhbnk7XG4gICAgICAgICAgICBpZihlbGVtID09IG51bGwpXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgbG9nLmVycm9yKCdwYXNzcGhyYXNlSW5wdXQgZWxlbWVudCByZWZlcmVuY2UgaXMgbnVsbCcpO1xuICAgICAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHBhc3NwaHJhc2UgPSBlbGVtLnZhbHVlO1xuICAgICAgICAgICAgaWYgKHBhc3NwaHJhc2UgPT0gJycgfHwgcGFzc3BocmFzZSA9PSBudWxsKSB7XG4gICAgICAgICAgICAgICAgLy8gICAgIC8vIGFsZXJ0KCdQbGVhc2UgaW5wdXQgcGFzc3BocmFzZScpXG4gICAgICAgICAgICAgICAgLy8gICAgIC8vIGFsZXJ0aWZ5LmVycm9yKCdQbGVhc2UgZW50ZXIgYSBwYXNzcGhyYXNlJyk7XG4gICAgICAgICAgICAgICAgbG9nLmVycm9yKCdwYXNzcGhyYXNlIGlzIGVtcHR5IG9yIG51bGwnKTtcbiAgICAgICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICAvLyBAdHMtaWdub3JlOiBPYmplY3QgaXMgcG9zc2libHkgJ251bGwnLlxuICAgICAgICAgICAgZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ25vLXVzZXItc2VsZWN0ZWQnKS5oaWRkZW4gPSB0cnVlO1xuICAgICAgICAgICAgLy8gQHRzLWlnbm9yZTogT2JqZWN0IGlzIHBvc3NpYmx5ICdudWxsJy5cbiAgICAgICAgICAgIGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdjaGF0LWNhcmQnKS5oaWRkZW4gPSBmYWxzZTtcbiAgICAgICAgICAgIC8vIEB0cy1pZ25vcmU6IE9iamVjdCBpcyBwb3NzaWJseSAnbnVsbCcuXG4gICAgICAgICAgICBlbGVtLmhpZGRlbiA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgICAgLy8gY29uc29sZS5sb2codGhpcy5nZXRFbGVtZW50c0J5Q2xhc3NOYW1lKCd0by11c2VyLXNwYW4nKSk7XG4gICAgICAgIGxldCBlbGVtID0gZWwuZ2V0RWxlbWVudHNCeUNsYXNzTmFtZSgndG8tdXNlci1zcGFuJylbMF0gYXMgSFRNTEVsZW1lbnQ7XG4gICAgICAgIGxldCB1c2VyTmFtZSA9IGVsZW0uaW5uZXJUZXh0O1xuICAgICAgICAvLyBAdHMtaWdub3JlOiBPYmplY3QgaXMgcG9zc2libHkgJ251bGwnLlxuICAgICAgICBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgndXNlci1uYW1lLXNwYW4nKS5pbm5lclRleHQgPSB1c2VyTmFtZTtcbiAgICAgICAgdGhpcy5fY2hhdE1vZGVsLmdldG1lc3NhZ2VzKHVzZXJOYW1lLCBwYXNzcGhyYXNlLCBudWxsKTtcbiAgICAgICAgLy8gcG9wdWxhdGVNZXNzYWdlcyh1c2VyTmFtZSwgcGFzc3BocmFzZSk7XG4gICAgICAgIHNlc3Npb25TdG9yYWdlLnNldEl0ZW0oJ3NlbGVjdGVkVXNlcicsIHVzZXJOYW1lKTtcbiAgICAgICAgZWwuY2xhc3NOYW1lICs9IFwiIGFjdGl2ZVwiO1xuICAgIH1cblxufSIsImV4cG9ydCBjbGFzcyBBY3RpdmVVc2VyVmlld01vZGVsIHtcbiAgICB1c2VyTmFtZTogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICAgIG9ubGluZTogYm9vbGVhbiB8IHVuZGVmaW5lZDtcbiAgICBsYXN0QWN0aXZlOiBzdHJpbmd8IHVuZGVmaW5lZDtcbn0iLCJleHBvcnQgY2xhc3MgQ2hhdE1lc3NhZ2VWaWV3TW9kZWwge1xuICAgIHB1YmxpYyB0b1VzZXI6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgICBwdWJsaWMgZnJvbVVzZXI6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgICBwdWJsaWMgbWVzc2FnZSE6IHN0cmluZztcbiAgICBwdWJsaWMgbWVzc2FnZVRpbWUhOiBEYXRlO1xuXG4gICAgXG59Il19 diff --git a/chatto/src/main/resources/static/js/login.js b/chatto/src/main/resources/static/js/login.js index 0cd8c80..7f67a51 100644 --- a/chatto/src/main/resources/static/js/login.js +++ b/chatto/src/main/resources/static/js/login.js @@ -3,7 +3,7 @@ function storeCredentials() { let passwordInput = document.getElementById('password'); var jqxhr = $.ajax({ type: 'GET', - url: `http://${hostAddress}/api/chat/get/token`, + url: `/api/chat/get/token`, headers: { "Authorization": "Basic " + btoa(usernameInput.value + ":" + passwordInput.value) }, @@ -28,7 +28,7 @@ function storeCredentials() { let secondClick = false; $('#loginForm').on('submit', function(e) { - if(!secondClick) { + if (!secondClick) { secondClick = true; e.preventDefault(); storeCredentials(); @@ -36,4 +36,4 @@ $('#loginForm').on('submit', function(e) { // else { // secondClick = false; // } -}) +}) \ No newline at end of file diff --git a/chatto/src/main/resources/static/js/scljs.js b/chatto/src/main/resources/static/js/scljs.js deleted file mode 100644 index c3c3ea9..0000000 --- a/chatto/src/main/resources/static/js/scljs.js +++ /dev/null @@ -1,60 +0,0 @@ -"use strict";var sjcl={cipher:{},hash:{},keyexchange:{},mode:{},misc:{},codec:{},exception:{corrupt:function(a){this.toString=function(){return"CORRUPT: "+this.message};this.message=a},invalid:function(a){this.toString=function(){return"INVALID: "+this.message};this.message=a},bug:function(a){this.toString=function(){return"BUG: "+this.message};this.message=a},notReady:function(a){this.toString=function(){return"NOT READY: "+this.message};this.message=a}}}; -sjcl.cipher.aes=function(a){this.s[0][0][0]||this.O();var b,c,d,e,f=this.s[0][4],g=this.s[1];b=a.length;var h=1;if(4!==b&&6!==b&&8!==b)throw new sjcl.exception.invalid("invalid aes key size");this.b=[d=a.slice(0),e=[]];for(a=b;a<4*b+28;a++){c=d[a-1];if(0===a%b||8===b&&4===a%b)c=f[c>>>24]<<24^f[c>>16&255]<<16^f[c>>8&255]<<8^f[c&255],0===a%b&&(c=c<<8^c>>>24^h<<24,h=h<<1^283*(h>>7));d[a]=d[a-b]^c}for(b=0;a;b++,a--)c=d[b&3?a:a-4],e[b]=4>=a||4>b?c:g[0][f[c>>>24]]^g[1][f[c>>16&255]]^g[2][f[c>>8&255]]^g[3][f[c& -255]]}; -sjcl.cipher.aes.prototype={encrypt:function(a){return t(this,a,0)},decrypt:function(a){return t(this,a,1)},s:[[[],[],[],[],[]],[[],[],[],[],[]]],O:function(){var a=this.s[0],b=this.s[1],c=a[4],d=b[4],e,f,g,h=[],k=[],l,n,m,p;for(e=0;0x100>e;e++)k[(h[e]=e<<1^283*(e>>7))^e]=e;for(f=g=0;!c[f];f^=l||1,g=k[g]||1)for(m=g^g<<1^g<<2^g<<3^g<<4,m=m>>8^m&255^99,c[f]=m,d[m]=f,n=h[e=h[l=h[f]]],p=0x1010101*n^0x10001*e^0x101*l^0x1010100*f,n=0x101*h[m]^0x1010100*m,e=0;4>e;e++)a[e][f]=n=n<<24^n>>>8,b[e][m]=p=p<<24^p>>>8;for(e= -0;5>e;e++)a[e]=a[e].slice(0),b[e]=b[e].slice(0)}}; -function t(a,b,c){if(4!==b.length)throw new sjcl.exception.invalid("invalid aes block size");var d=a.b[c],e=b[0]^d[0],f=b[c?3:1]^d[1],g=b[2]^d[2];b=b[c?1:3]^d[3];var h,k,l,n=d.length/4-2,m,p=4,r=[0,0,0,0];h=a.s[c];a=h[0];var q=h[1],v=h[2],w=h[3],x=h[4];for(m=0;m>>24]^q[f>>16&255]^v[g>>8&255]^w[b&255]^d[p],k=a[f>>>24]^q[g>>16&255]^v[b>>8&255]^w[e&255]^d[p+1],l=a[g>>>24]^q[b>>16&255]^v[e>>8&255]^w[f&255]^d[p+2],b=a[b>>>24]^q[e>>16&255]^v[f>>8&255]^w[g&255]^d[p+3],p+=4,e=h,f=k,g=l;for(m= -0;4>m;m++)r[c?3&-m:m]=x[e>>>24]<<24^x[f>>16&255]<<16^x[g>>8&255]<<8^x[b&255]^d[p++],h=e,e=f,f=g,g=b,b=h;return r} -sjcl.bitArray={bitSlice:function(a,b,c){a=sjcl.bitArray.$(a.slice(b/32),32-(b&31)).slice(1);return void 0===c?a:sjcl.bitArray.clamp(a,c-b)},extract:function(a,b,c){var d=Math.floor(-b-c&31);return((b+c-1^b)&-32?a[b/32|0]<<32-d^a[b/32+1|0]>>>d:a[b/32|0]>>>d)&(1<>b-1,1));return a},partial:function(a,b,c){return 32===a?b:(c?b|0:b<<32-a)+0x10000000000*a},getPartial:function(a){return Math.round(a/0x10000000000)||32},equal:function(a,b){if(sjcl.bitArray.bitLength(a)!==sjcl.bitArray.bitLength(b))return!1;var c=0,d;for(d=0;d>>b),c=a[e]<<32-b;e=a.length?a[a.length-1]:0;a=sjcl.bitArray.getPartial(e);d.push(sjcl.bitArray.partial(b+a&31,32>>24|c>>>8&0xff00|(c&0xff00)<<8|c<<24;return a}}; -sjcl.codec.utf8String={fromBits:function(a){var b="",c=sjcl.bitArray.bitLength(a),d,e;for(d=0;d>>8>>>8>>>8),e<<=8;return decodeURIComponent(escape(b))},toBits:function(a){a=unescape(encodeURIComponent(a));var b=[],c,d=0;for(c=0;c>>g)>>>e),gn){if(!b)try{return sjcl.codec.base32hex.toBits(a)}catch(p){}throw new sjcl.exception.invalid("this isn't "+m+"!");}h>e?(h-=e,f.push(l^n>>>h),l=n<>>e)>>>26),6>e?(g=a[c]<<6-e,e+=26,c++):(g<<=6,e-=6);for(;d.length&3&&!b;)d+="=";return d},toBits:function(a,b){a=a.replace(/\s|=/g,"");var c=[],d,e=0,f=sjcl.codec.base64.B,g=0,h;b&&(f=f.substr(0,62)+"-_");for(d=0;dh)throw new sjcl.exception.invalid("this isn't base64!");26>>e),g=h<<32-e):(e+=6,g^=h<<32-e)}e&56&&c.push(sjcl.bitArray.partial(e&56,g,1));return c}};sjcl.codec.base64url={fromBits:function(a){return sjcl.codec.base64.fromBits(a,1,1)},toBits:function(a){return sjcl.codec.base64.toBits(a,1)}};sjcl.hash.sha256=function(a){this.b[0]||this.O();a?(this.F=a.F.slice(0),this.A=a.A.slice(0),this.l=a.l):this.reset()};sjcl.hash.sha256.hash=function(a){return(new sjcl.hash.sha256).update(a).finalize()}; -sjcl.hash.sha256.prototype={blockSize:512,reset:function(){this.F=this.Y.slice(0);this.A=[];this.l=0;return this},update:function(a){"string"===typeof a&&(a=sjcl.codec.utf8String.toBits(a));var b,c=this.A=sjcl.bitArray.concat(this.A,a);b=this.l;a=this.l=b+sjcl.bitArray.bitLength(a);if(0x1fffffffffffffb;c++){e=!0;for(d=2;d*d<=c;d++)if(0===c%d){e= -!1;break}e&&(8>b&&(this.Y[b]=a(Math.pow(c,.5))),this.b[b]=a(Math.pow(c,1/3)),b++)}}}; -function u(a,b){var c,d,e,f=a.F,g=a.b,h=f[0],k=f[1],l=f[2],n=f[3],m=f[4],p=f[5],r=f[6],q=f[7];for(c=0;64>c;c++)16>c?d=b[c]:(d=b[c+1&15],e=b[c+14&15],d=b[c&15]=(d>>>7^d>>>18^d>>>3^d<<25^d<<14)+(e>>>17^e>>>19^e>>>10^e<<15^e<<13)+b[c&15]+b[c+9&15]|0),d=d+q+(m>>>6^m>>>11^m>>>25^m<<26^m<<21^m<<7)+(r^m&(p^r))+g[c],q=r,r=p,p=m,m=n+d|0,n=l,l=k,k=h,h=d+(k&l^n&(k^l))+(k>>>2^k>>>13^k>>>22^k<<30^k<<19^k<<10)|0;f[0]=f[0]+h|0;f[1]=f[1]+k|0;f[2]=f[2]+l|0;f[3]=f[3]+n|0;f[4]=f[4]+m|0;f[5]=f[5]+p|0;f[6]=f[6]+r|0;f[7]= -f[7]+q|0} -sjcl.mode.ccm={name:"ccm",G:[],listenProgress:function(a){sjcl.mode.ccm.G.push(a)},unListenProgress:function(a){a=sjcl.mode.ccm.G.indexOf(a);-1k)throw new sjcl.exception.invalid("ccm: iv must be at least 7 bytes");for(f=2;4>f&&l>>>8*f;f++);f<15-k&&(f=15-k);c=h.clamp(c, -8*(15-f));b=sjcl.mode.ccm.V(a,b,c,d,e,f);g=sjcl.mode.ccm.C(a,g,c,b,e,f);return h.concat(g.data,g.tag)},decrypt:function(a,b,c,d,e){e=e||64;d=d||[];var f=sjcl.bitArray,g=f.bitLength(c)/8,h=f.bitLength(b),k=f.clamp(b,h-e),l=f.bitSlice(b,h-e),h=(h-e)/8;if(7>g)throw new sjcl.exception.invalid("ccm: iv must be at least 7 bytes");for(b=2;4>b&&h>>>8*b;b++);b<15-g&&(b=15-g);c=f.clamp(c,8*(15-b));k=sjcl.mode.ccm.C(a,k,c,l,e,b);a=sjcl.mode.ccm.V(a,k.data,c,d,e,b);if(!f.equal(k.tag,a))throw new sjcl.exception.corrupt("ccm: tag doesn't match"); -return k.data},na:function(a,b,c,d,e,f){var g=[],h=sjcl.bitArray,k=h.i;d=[h.partial(8,(b.length?64:0)|d-2<<2|f-1)];d=h.concat(d,c);d[3]|=e;d=a.encrypt(d);if(b.length)for(c=h.bitLength(b)/8,65279>=c?g=[h.partial(16,c)]:0xffffffff>=c&&(g=h.concat([h.partial(16,65534)],[c])),g=h.concat(g,b),b=0;be||16n&&(sjcl.mode.ccm.fa(g/ -k),n+=m),c[3]++,e=a.encrypt(c),b[g]^=e[0],b[g+1]^=e[1],b[g+2]^=e[2],b[g+3]^=e[3];return{tag:d,data:h.clamp(b,l)}}}; -sjcl.mode.ocb2={name:"ocb2",encrypt:function(a,b,c,d,e,f){if(128!==sjcl.bitArray.bitLength(c))throw new sjcl.exception.invalid("ocb iv must be 128 bits");var g,h=sjcl.mode.ocb2.S,k=sjcl.bitArray,l=k.i,n=[0,0,0,0];c=h(a.encrypt(c));var m,p=[];d=d||[];e=e||64;for(g=0;g+4e.bitLength(c)&&(h=f(h,d(h)),c=e.concat(c,[-2147483648,0,0,0]));g=f(g,c); -return a.encrypt(f(d(f(h,d(h))),g))},S:function(a){return[a[0]<<1^a[1]>>>31,a[1]<<1^a[2]>>>31,a[2]<<1^a[3]>>>31,a[3]<<1^135*(a[0]>>>31)]}}; -sjcl.mode.gcm={name:"gcm",encrypt:function(a,b,c,d,e){var f=b.slice(0);b=sjcl.bitArray;d=d||[];a=sjcl.mode.gcm.C(!0,a,f,d,c,e||128);return b.concat(a.data,a.tag)},decrypt:function(a,b,c,d,e){var f=b.slice(0),g=sjcl.bitArray,h=g.bitLength(f);e=e||128;d=d||[];e<=h?(b=g.bitSlice(f,h-e),f=g.bitSlice(f,0,h-e)):(b=f,f=[]);a=sjcl.mode.gcm.C(!1,a,f,d,c,e);if(!g.equal(a.tag,b))throw new sjcl.exception.corrupt("gcm: tag doesn't match");return a.data},ka:function(a,b){var c,d,e,f,g,h=sjcl.bitArray.i;e=[0,0, -0,0];f=b.slice(0);for(c=0;128>c;c++){(d=0!==(a[Math.floor(c/32)]&1<<31-c%32))&&(e=h(e,f));g=0!==(f[3]&1);for(d=3;0>>1|(f[d-1]&1)<<31;f[0]>>>=1;g&&(f[0]^=-0x1f000000)}return e},j:function(a,b,c){var d,e=c.length;b=b.slice(0);for(d=0;de&&(a=b.hash(a));for(d=0;dd||0>c)throw new sjcl.exception.invalid("invalid params to pbkdf2");"string"===typeof a&&(a=sjcl.codec.utf8String.toBits(a));"string"===typeof b&&(b=sjcl.codec.utf8String.toBits(b));e=e||sjcl.misc.hmac;a=new e(a);var f,g,h,k,l=[],n=sjcl.bitArray;for(k=1;32*l.length<(d||1);k++){e=f=a.encrypt(n.concat(b,[k]));for(g=1;gg;g++)e.push(0x100000000*Math.random()|0);for(g=0;g=1<this.o&&(this.o= -f);this.P++;this.b=sjcl.hash.sha256.hash(this.b.concat(e));this.L=new sjcl.cipher.aes(this.b);for(d=0;4>d&&(this.h[d]=this.h[d]+1|0,!this.h[d]);d++);}for(d=0;d>>1;this.c[g].update([d,this.N++,2,b,f,a.length].concat(a))}break;case "string":void 0===b&&(b=a.length);this.c[g].update([d,this.N++,3,b,f,a.length]);this.c[g].update(a);break;default:k=1}if(k)throw new sjcl.exception.bug("random: addEntropy only supports number, array of numbers or string");this.m[g]+=b;this.f+=b;h===this.u&&(this.isReady()!==this.u&&A("seeded",Math.max(this.o,this.f)),A("progress",this.getProgress()))}, -isReady:function(a){a=this.T[void 0!==a?a:this.M];return this.o&&this.o>=a?this.m[0]>this.ba&&(new Date).valueOf()>this.Z?this.J|this.I:this.I:this.f>=a?this.J|this.u:this.u},getProgress:function(a){a=this.T[a?a:this.M];return this.o>=a?1:this.f>a?1:this.f/a},startCollectors:function(){if(!this.D){this.a={loadTimeCollector:B(this,this.ma),mouseCollector:B(this,this.oa),keyboardCollector:B(this,this.la),accelerometerCollector:B(this,this.ea),touchCollector:B(this,this.qa)};if(window.addEventListener)window.addEventListener("load", -this.a.loadTimeCollector,!1),window.addEventListener("mousemove",this.a.mouseCollector,!1),window.addEventListener("keypress",this.a.keyboardCollector,!1),window.addEventListener("devicemotion",this.a.accelerometerCollector,!1),window.addEventListener("touchmove",this.a.touchCollector,!1);else if(document.attachEvent)document.attachEvent("onload",this.a.loadTimeCollector),document.attachEvent("onmousemove",this.a.mouseCollector),document.attachEvent("keypress",this.a.keyboardCollector);else throw new sjcl.exception.bug("can't attach event"); -this.D=!0}},stopCollectors:function(){this.D&&(window.removeEventListener?(window.removeEventListener("load",this.a.loadTimeCollector,!1),window.removeEventListener("mousemove",this.a.mouseCollector,!1),window.removeEventListener("keypress",this.a.keyboardCollector,!1),window.removeEventListener("devicemotion",this.a.accelerometerCollector,!1),window.removeEventListener("touchmove",this.a.touchCollector,!1)):document.detachEvent&&(document.detachEvent("onload",this.a.loadTimeCollector),document.detachEvent("onmousemove", -this.a.mouseCollector),document.detachEvent("keypress",this.a.keyboardCollector)),this.D=!1)},addEventListener:function(a,b){this.K[a][this.ga++]=b},removeEventListener:function(a,b){var c,d,e=this.K[a],f=[];for(d in e)e.hasOwnProperty(d)&&e[d]===b&&f.push(d);for(c=0;cb&&(a.h[b]=a.h[b]+1|0,!a.h[b]);b++);return a.L.encrypt(a.h)} -function B(a,b){return function(){b.apply(a,arguments)}}sjcl.random=new sjcl.prng(6); -a:try{var D,E,F,G;if(G="undefined"!==typeof module&&module.exports){var H;try{H=require("crypto")}catch(a){H=null}G=E=H}if(G&&E.randomBytes)D=E.randomBytes(128),D=new Uint32Array((new Uint8Array(D)).buffer),sjcl.random.addEntropy(D,1024,"crypto['randomBytes']");else if("undefined"!==typeof window&&"undefined"!==typeof Uint32Array){F=new Uint32Array(32);if(window.crypto&&window.crypto.getRandomValues)window.crypto.getRandomValues(F);else if(window.msCrypto&&window.msCrypto.getRandomValues)window.msCrypto.getRandomValues(F); -else break a;sjcl.random.addEntropy(F,1024,"crypto['getRandomValues']")}}catch(a){"undefined"!==typeof window&&window.console&&(console.log("There was an error collecting entropy from the browser:"),console.log(a))} -sjcl.json={defaults:{v:1,iter:1E4,ks:128,ts:64,mode:"ccm",adata:"",cipher:"aes"},ja:function(a,b,c,d){c=c||{};d=d||{};var e=sjcl.json,f=e.g({iv:sjcl.random.randomWords(4,0)},e.defaults),g;e.g(f,c);c=f.adata;"string"===typeof f.salt&&(f.salt=sjcl.codec.base64.toBits(f.salt));"string"===typeof f.iv&&(f.iv=sjcl.codec.base64.toBits(f.iv));if(!sjcl.mode[f.mode]||!sjcl.cipher[f.cipher]||"string"===typeof a&&100>=f.iter||64!==f.ts&&96!==f.ts&&128!==f.ts||128!==f.ks&&192!==f.ks&&0x100!==f.ks||2>f.iv.length|| -4=b.iter||64!==b.ts&&96!==b.ts&&128!==b.ts||128!==b.ks&&192!==b.ks&&0x100!==b.ks||!b.iv||2>b.iv.length||4 Chat + + + + - - - @@ -149,7 +159,7 @@ -
+