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,{"version":3,"sources":["../../../../../../../usr/local/lib/node_modules/watchify/node_modules/browser-pack/_prelude.js","ts/src/controller/ChatController.ts","ts/src/controller/UserController.ts","ts/src/main.ts","ts/src/model/ChatModel.ts","ts/src/model/ChatModelHelper.ts","ts/src/model/FetchErrorHandler.ts","ts/src/model/UserModel.ts","ts/src/service/SJCLEncryptionService.ts","ts/src/singleton/JsonAPI.ts","ts/src/template/TemplateFactory.ts","ts/src/view/ChatView.ts","ts/src/view/UserView.ts","ts/src/viewmodel/ActiveUserViewModel.ts","ts/src/viewmodel/ChatMessageViewModel.ts"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(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<t.length;i++)o(t[i]);return o}return r})()","import { Controller } from \"./AbstractController\";\nimport \"../model/AbstractModel\"\nimport \"../model/UserModel\"\nimport \"../view/AbstractView\"\nimport \"../view/UserView\"\nimport { Model } from \"../model/AbstractModel\";\nimport { View } from \"../view/AbstractView\";\nimport { ChatMessageViewModel } from \"../viewmodel/ChatMessageViewModel\";\nimport { ChatModel } from \"../model/ChatModel\";\nimport { ChatView } from \"../view/ChatView\";\n\nexport class ChatController {\n    private _model: ChatModel;\n    private _view: ChatView;\n\n\n    constructor(model: ChatModel, view: ChatView) {\n        this._model = model;\n        this._view = view;\n    }\n\n\n    /**\n     * eventHandler\n     */\n    public eventHandler(vm: ChatMessageViewModel[]): void {\n        this._model.someBusinessMethod(vm);\n    }\n\n    public test(): void {\n        const chatMessageViewModels: ChatMessageViewModel[] = [];\n        let chatMessageViewModelMock = new ChatMessageViewModel();\n        chatMessageViewModelMock.fromUser = \"user1\";\n        chatMessageViewModelMock.toUser = \"user2\";\n        chatMessageViewModelMock.message = \"\";\n        chatMessageViewModelMock.messageTime = new Date();\n        chatMessageViewModels.push(chatMessageViewModelMock);\n    }\n\n    \n}","import { Controller } from \"./AbstractController\";\nimport \"../model/AbstractModel\"\nimport \"../model/UserModel\"\nimport \"../view/AbstractView\"\nimport \"../view/UserView\"\nimport { Model } from \"../model/AbstractModel\";\nimport { View } from \"../view/AbstractView\";\nimport { ActiveUserViewModel } from \"../viewmodel/ActiveUserViewModel\";\nimport { UserView } from \"../view/UserView\";\nimport { UserModel } from \"../model/UserModel\";\n\nexport class UserController {\n    private _model: UserModel;\n    private _view: UserView;\n\n\n    constructor(model: UserModel, view: UserView) {\n        this._model = model;\n        this._view = view;\n    }\n\n\n    /**\n     * eventHandler\n     */\n    public eventHandler(vm: ActiveUserViewModel[]): void {\n        this._model.someBusinessMethod(vm);\n    }\n\n    public test(): void {\n        const activeUsersMock: ActiveUserViewModel[] = [];\n        let activeUserViewModelMock = new ActiveUserViewModel();\n        activeUserViewModelMock.userName = \"some user\";\n        activeUserViewModelMock.lastActive = \"3 hrs ago\";\n        activeUserViewModelMock.online = true;\n        activeUsersMock.push(activeUserViewModelMock);\n        activeUserViewModelMock = new ActiveUserViewModel();\n        activeUserViewModelMock.lastActive = \"3 hrs ago\";\n        activeUserViewModelMock.online = true;\n        activeUserViewModelMock.userName = \"some user 2\";\n        activeUsersMock.push(activeUserViewModelMock);\n        this.eventHandler(activeUsersMock);\n    }\n\n    public getActiveUsers(): void {\n        this._model.getActiveUsers();\n    }\n\n    \n}","import { Controller } from \"./controller/AbstractController\";\nimport { UserModel } from \"./model/UserModel\"\nimport { Model } from \"./model/AbstractModel\";\nimport { View } from \"./view/AbstractView\";\nimport { UserView } from \"./view/UserView\";\nimport { UserController } from \"./controller/UserController\";\nimport { ModelFactory } from \"./model/ModelFactory\";\nimport { ActiveUserViewModel } from \"./viewmodel/ActiveUserViewModel\";\nimport { ChatMessageViewModel } from \"./viewmodel/ChatMessageViewModel\";\nimport * as Handlebars from \"handlebars\";\nimport markdownit = require('markdown-it');\nimport { ChatModel } from \"./model/ChatModel\";\nimport { ChatView } from \"./view/ChatView\";\nimport { ChatController } from \"./controller/ChatController\";\nimport { JsonAPI } from \"./singleton/JsonAPI\";\n// import log = require('loglevel')\nimport * as log from 'loglevel';\n// import log from 'loglevel';\nimport { EncryptionService } from \"./service/EncryptionService\";\nimport { SJCLEncryptionService } from \"./service/SJCLEncryptionService\";\n// var markdownit = require('markdown-it');\nvar md = new markdownit();\n\n\nconst userBox = document.getElementById('contacts-box');\n\nlog.setLevel(\"TRACE\")\n\nconst chatModel = new ChatModel();\n\nconst userModel = new UserModel();\n// const userModel = ModelFactory.createModel(\"USER\");\n// @ts-ignore: Argument of type 'HTMLElement | null' is not assignable to parameter of type 'HTMLElement'. Type 'null' is not assignable to type 'HTMLElement'.\nconst userView = new UserView(userModel, chatModel, userBox);\n\n// console.log(userBox);\n\nuserModel.attach(userView);\n// userView.model\n\n\nconst userController = new UserController(userModel, userView);\n// userController.test();\n\n\n\n// userModel.someBusinessMethod(activeUsersMock);\nlog.info(\"hello\");\n\nconst chatArea = document.getElementById('chat-area-new');\n// @ts-ignore: Argument of type 'HTMLElement | null' is not assignable to parameter of type 'HTMLElement'. Type 'null' is not assignable to type 'HTMLElement'.\nconst chatView = new ChatView(chatModel, chatArea);\nchatModel.attach(chatView);\nconst chatController = new ChatController(chatModel, chatView);\n\n\nfunction someFunc(vm: ActiveUserViewModel): void {\n    // log.info(vm);\n    // logger.info(vm)\n}\n\nuserController.getActiveUsers();\n\nlog.info(\"test\");\n// someFunc(activeUserViewModelMock);\n\n// @ts-ignore: Object is possibly 'null'.\nvar source = document.getElementById(\"msg_container_template\").innerHTML;\n\nvar msgContainerTemplate = Handlebars.compile(source);\n\nJsonAPI.ACTIVE_USERS_GET + 'aef';\n\nconst encryptionService: EncryptionService = new SJCLEncryptionService();\nlet ct = encryptionService.encrypt(\"password\",\"data\");\nconsole.log(encryptionService.decrypt(\"password\", JSON.parse(ct as string)));\n\nHandlebars.registerHelper('avatar', function() {\n    return '<div class=\"img_cont_msg\"> <img src=\"https://static.turbosquid.com/Preview/001292/481/WV/_D.jpg\" class=\"rounded-circle user_img_msg\"> </div>';\n});","import { Subject } from \"../observe/Observable\";\nimport { Model } from \"./AbstractModel\";\nimport { Observer } from \"../observe/Observer\";\nimport { fetchErrorHandler } from \"./FetchErrorHandler\";\nimport { ActiveUserViewModel } from \"../viewmodel/ActiveUserViewModel\";\nimport { ChatMessageViewModel } from \"../viewmodel/ChatMessageViewModel\";\nimport { JsonAPI } from \"../singleton/JsonAPI\";\nimport log = require('loglevel');\nimport { EncryptionService } from \"../service/EncryptionService\";\nimport { SJCLEncryptionService } from \"../service/SJCLEncryptionService\";\nimport { ChatMessageDTO } from \"../dto/ChatMessageDTO\";\nimport { ChatModelHelper } from \"./ChatModelHelper\";\n\nexport class ChatModel implements Subject {\n    /**\n  * @type {Observer[]} List of subscribers. In real life, the list of\n  * subscribers can be stored more comprehensively (categorized by event\n  * type, etc.).\n  */\n    private readonly _observers: Observer[] = [];\n    private state: ChatMessageViewModel[] | null;\n    private readonly _messagesMap: Map<string, ChatMessageViewModel[]>;\n\n    constructor() {\n        this.state = null;\n        this._messagesMap = new Map();\n    }\n    /**\n     * The subscription management methods.\n     */\n    public attach(observer: Observer): void {\n        console.log('Subject: Attached an observer.');\n        this._observers.push(observer);\n    }\n\n    public detach(observer: Observer): void {\n        const observerIndex = this._observers.indexOf(observer);\n        this._observers.splice(observerIndex, 1);\n        console.log('Subject: Detached an observer.');\n    }\n\n    public setUserMessages(username: string, messages: ChatMessageViewModel[]) {\n        this._messagesMap.set(username, messages);\n    }\n\n    /**\n     * Trigger an update in each subscriber.\n     */\n    public notify(userName: string): void {\n        console.log('Subject: Notifying observers...');\n        for (const observer of this._observers) {\n            observer.update(this._messagesMap.get(userName));\n        }\n    }\n\n    public someBusinessMethod(chatMessageList: ChatMessageViewModel[]): void {\n        this.state = chatMessageList;\n        this.helperMethod();\n        console.log(`Subject: My state has just changed`);\n        console.log(chatMessageList);\n        this.notify(\"some user\");\n    }\n\n    public async getmessages(userName: string, passphrase: string, lastMessageTime: string | null): Promise<ChatMessageViewModel[]> {\n        const cVMs = await ChatModelHelper.getMessages(userName, passphrase, lastMessageTime, this);\n        if (cVMs != null) {\n            log.info('Subject: My state has just changed')\n            log.debug(cVMs);\n            this._messagesMap.set(userName, cVMs);\n            this.notify(userName);\n        }\n        else {\n            log.error('Messages were null');\n        }\n\n        return cVMs;\n    }\n\n\n\n    private helperMethod() { }\n\n    public populateMessages(): void {\n\n    }\n\n\n\n}","import { ChatMessageDTO } from \"../dto/ChatMessageDTO\";\n\nimport * as log from \"loglevel\";\n\nimport { ChatMessageViewModel } from \"../viewmodel/ChatMessageViewModel\";\n\nimport { JsonAPI } from \"../singleton/JsonAPI\";\n\nimport { fetchErrorHandler } from \"./FetchErrorHandler\";\n\nimport { EncryptionService } from \"../service/EncryptionService\";\nimport { SJCLEncryptionService } from \"../service/SJCLEncryptionService\";\nimport { ChatModel } from \"./ChatModel\"\nexport class ChatModelHelper {\n    private static readonly encryptionService = new SJCLEncryptionService();\n\n    public static async getMessages(userName: string, passphrase: string, lastMessageTime: string | null, chatModel: ChatModel): Promise<ChatMessageViewModel[]> {\n        switch (lastMessageTime) {\n            case null: {\n                // this.getAllMessagesAjax(userName)\n                //     .then((data: ChatMessageDTO[]) => {\n                //         log.debug(`Subject: received all messages`);\n                //         // let userNames = data.map(ChatMessageViewModel => ChatMessageViewModel.fromUser)\n                //         // let sumt = data.map(chatMessageViewModel => { return this.encryptionService.decrypt(passphrase, chatMessageViewModel.messageCipher) });\n                //         return data.map(vm => this.toChatMessageVM(vm, passphrase));\n                //         // chatModel.setUserMessages(userName, chatMessageVMs);\n                //         // chatModel.notify();\n                //     })\n                // break;\n\n                const data: ChatMessageDTO[] = await this.getAllMessagesAjax(userName);\n                return data.map(vm => this.toChatMessageVM(vm, passphrase));\n\n            }\n            default: {\n                // this.getNewMessagesAjax(userName, lastMessageTime)\n                //     .then((data: ChatMessageDTO[]) => {\n                //         log.debug(`Subject: received new messages`);\n                //         return data.map(vm => this.toChatMessageVM(vm, passphrase));\n                //         // chatModel.setUserMessages(userName, chatMessageVMs);\n                //         // this.state = data;\n                //         // chatModel.notify();\n                //     })\n                // break;\n                const data: ChatMessageDTO[] = await this.getNewMessagesAjax(userName, lastMessageTime);\n                return data.map(vm => this.toChatMessageVM(vm, passphrase));\n            }\n        }\n\n        // return null;\n    }\n\n    private static toChatMessageVM(chatMessageDTO: ChatMessageDTO, passphrase: string): ChatMessageViewModel {\n        const vm = new ChatMessageViewModel();\n        vm.fromUser = chatMessageDTO.fromUser;\n        vm.toUser = chatMessageDTO.toUser;\n        vm.messageTime = chatMessageDTO.messageTime;\n        vm.message = this.encryptionService.decrypt(passphrase, chatMessageDTO.messageCipher) as string;\n        return vm;\n    }\n\n    private static async getAllMessagesAjax(toUser: string): Promise<any> {\n        const headers = new Headers();\n        if (JsonAPI.authToken == null) {\n            log.error(\"authToken null\");\n            return;\n        };\n        headers.append('X-AUTH-TOKEN', JsonAPI.authToken);\n        const response = await fetch(`${JsonAPI.CHAT_MESSAGES_GET}/${toUser}`, {\n            method: 'GET',\n            headers: headers\n        });\n        console.log(response.clone());\n        if (fetchErrorHandler(response.clone())) {\n            return null;\n        }\n        const data: Promise<any> = await response.json();\n        return data;\n    }\n\n    private static async getNewMessagesAjax(toUser: string, lastMessageTimeStamp: string): Promise<any> {\n        const headers = new Headers();\n        if (JsonAPI.authToken == null) {\n            log.error(\"authToken null\");\n            return;\n        };\n        headers.append('X-AUTH-TOKEN', JsonAPI.authToken);\n        const response = await fetch(`${JsonAPI.CHAT_MESSAGES_GET}/${toUser}/${lastMessageTimeStamp}`, {\n            method: 'GET',\n            headers: headers\n        });\n        console.log(response.clone());\n        if (fetchErrorHandler(response.clone())) {\n            return null;\n        }\n        const data: Promise<any> = await response.json();\n        return data;\n    }\n}","import log = require(\"loglevel\");\nimport { sprintf } from \"sprintf-js\";\n// import sprintf = require('sprintf-js').sprintf;\n\nexport function fetchErrorHandler(response: Response) {\n    // alertify.success('Current position : ' + alertify.get('notifier', 'position'));\n    if (!response.ok) {\n        return response.text().catch(err => {\n            // the status was not ok and there is no json body\n            // throw new Error(response.statusText);\n            // window.alert(sprintf('Some error occured. Http code is %s', response.status));\n            // alertify.error(sprintf('Some error occured. Http code is %s', response.status));\n            log.error(sprintf('Some error occured. Http code is %s', response.status));\n            log.error();\n            return true;\n        }).then(json => {\n            // the status was not ok but there is a json body\n            // throw new Error(json.error.message); // example error message returned by a REST API\n            // window.alert(sprintf('Error: %s (Http code %s)', json, response.status));\n            // alertify.error(sprintf('Some error occured. Http code is %s', response.status));\n            log.error(sprintf('Some error occured. Http code is %s', response.status));\n            log.error(json);\n            return true;\n        });\n    }\n}","import { Subject } from \"../observe/Observable\";\nimport { Model } from \"./AbstractModel\";\nimport { Observer } from \"../observe/Observer\";\nimport { fetchErrorHandler } from \"./FetchErrorHandler\";\nimport { ActiveUserViewModel } from \"../viewmodel/ActiveUserViewModel\";\nimport { JsonAPI } from \"../singleton/JsonAPI\";\nimport { ChatMessageViewModel } from \"../viewmodel/ChatMessageViewModel\";\nimport * as log from \"loglevel\";\n\nexport class UserModel implements Subject {\n    /**\n  * @type {Observer[]} List of subscribers. In real life, the list of\n  * subscribers can be stored more comprehensively (categorized by event\n  * type, etc.).\n  */\n    private readonly observers: Observer[] = [];\n    private state: ActiveUserViewModel[] | undefined;\n    // @ts-ignore: Cannot find name 'hostAddress'.\n\n    constructor() { }\n    /**\n     * The subscription management methods.\n     */\n    public attach(observer: Observer): void {\n        console.log('Subject: Attached an observer.');\n        this.observers.push(observer);\n    }\n\n    public detach(observer: Observer): void {\n        const observerIndex = this.observers.indexOf(observer);\n        this.observers.splice(observerIndex, 1);\n        console.log('Subject: Detached an observer.');\n    }\n\n    /**\n     * Trigger an update in each subscriber.\n     */\n    public notify(): void {\n        console.log('Subject: Notifying observers...');\n        for (const observer of this.observers) {\n            observer.update(this.state);\n        }\n    }\n\n    public someBusinessMethod(activeuserList: ActiveUserViewModel[]): void {\n        this.state = activeuserList;\n        this.helperMethod();\n        console.log(`Subject: My state has just changed`);\n        console.log(activeuserList);\n        this.notify();\n    }\n\n    /**\n     * getActiveUsers\n     */\n    public getActiveUsers(): void {\n        if(JsonAPI.authToken!= null){\n        this.getActiveUsersAjax(JsonAPI.authToken, JsonAPI.ACTIVE_USERS_GET)\n            .then(data => {\n                // // activeUsers = data;\n                // sessionStorage.setItem('activeUsers', JSON.stringify(data));\n                // console.log(sessionStorage.getItem('activeUsers'));\n                console.log(`Subject: received ajax active users`);\n                this.state = data;\n                this.notify();\n            })\n        }\n        else {\n            log.error('Auth token is null');\n        }\n    }\n\n    async getActiveUsersAjax(authToken2: string, URL: string): Promise<any> {\n        let headers = new Headers();\n        // headers.append('Authorization', basicAuthToken);\n        headers.append('X-AUTH-TOKEN', authToken2);\n        let response = await fetch(JsonAPI.ACTIVE_USERS_GET, {\n            method: 'GET',\n            headers: headers\n        });\n        console.log(response.clone());\n        if (fetchErrorHandler(response.clone())) {\n            return null;\n        }\n        let data = await response.json();\n        // return data;\n        return new Promise((resolve, reject) => {\n            if (data != null)\n                resolve(data)\n            else\n                reject('Response data null')\n        })\n    }\n\n    private helperMethod() { }\n\n}","import { EncryptionService } from \"./EncryptionService\";\nimport * as sjcl from \"sjcl\";\n\nexport class SJCLEncryptionService implements EncryptionService {\n    private readonly params: any = { mode: \"gcm\", ts: 128, adata: \"\", iter: 10000}\n    public encrypt(passphrase: string, plainText: string): Object {\n        return sjcl.encrypt(passphrase, plainText, this.params);\n    } \n    \n    public decrypt(passphrase: string, cipher: Object): Object {\n        // return sjcl.decrypt(passphrase, cipher as sjcl.SjclCipherEncrypted, undefined, undefined);\n        return sjcl.decrypt(passphrase, JSON.stringify(cipher), undefined, undefined);\n    }\n}","export namespace JsonAPI {\n    // @ts-ignore: Cannot find name 'hostAddress'.\n    export let userName: string | null = localStorage.getItem('userName');\n    export let authToken: string | null = localStorage.getItem('authToken');\n    export const ACTIVE_USERS_GET = `/api/chat/get/active-users`;\n    export const CHAT_MESSAGES_GET = `/api/chat/get/messages`;\n    \n}","export class TemplateFactory {\n    // static getTemplate(templateName: string): Handlebars.TemplateDelegate<any> {\n\n    //     switch (templateName) {\n    //         case \"user-contact-online-template\": {\n    //             // let source = document.getElementById(\"user-contact-online-template\").innerHTML;\n    //             // let msgContainerTemplate = Handlebars.compile(source);\n    //             // return msgContainerTemplate;\n    //             return this.createTemplate(templateName);\n    //         }\n    //         case \"msg_container_send_template\": {\n    //             // let source = document.getElementById(\"msg_container_send_template\").innerHTML;\n    //             // let msgContainerTemplate = Handlebars.compile(source);\n    //             // return msgContainerTemplate;\n    //             return this.createTemplate(templateName);\n    //         }\n    //         case \"msg_container_template\": {\n    //             // let source = document.getElementById(\"msg_container_send_template\").innerHTML;\n    //             // let msgContainerTemplate = Handlebars.compile(source);\n    //             // return msgContainerTemplate;\n    //             return this.createTemplate(templateName);\n    //         }\n    //         default:\n    //             throw new Error('invalid template name');\n    //     }\n    // }\n\n    static getTemplate(templateName: string): Handlebars.TemplateDelegate<any> {\n        //@ts-ignore: Object is possibly 'null'.\n        let source = document.getElementById(templateName).innerHTML;\n        let msgContainerTemplate = Handlebars.compile(source);\n        return msgContainerTemplate;\n    }\n}","import { Observer } from \"../observe/Observer\";\nimport { Model } from \"../model/AbstractModel\";\nimport { Subject } from \"../observe/Observable\";\nimport { View } from \"./AbstractView\";\nimport { Controller } from \"../controller/AbstractController\";\nimport { TemplateFactory } from \"../template/TemplateFactory\";\nimport { ActiveUserViewModel } from \"../viewmodel/ActiveUserViewModel\";\nimport { ChatModel } from \"../model/ChatModel\";\nimport { ChatMessageViewModel } from \"../viewmodel/ChatMessageViewModel\";\nimport * as log from 'loglevel';\nimport * as DOMPurify from 'dompurify';\nimport markdownit = require('markdown-it');\nvar md = new markdownit();\n\nexport class ChatView implements Observer {\n    private readonly _model: ChatModel;\n    private readonly _element: HTMLElement;\n    private readonly _messageSendTemplate = TemplateFactory.getTemplate('msg_container_send_template');\n\n\n    constructor(model: ChatModel, element: HTMLElement) {\n        this._model = model;\n        this._element = element;\n    }\n\n\n    update(data: ChatMessageViewModel[]): void {\n        log.info('ChatView: updating view');\n        let html: string = \"\";\n        data.forEach((vm: ChatMessageViewModel) => {\n            html += this._messageSendTemplate(vm);\n        });\n\n        /** Very Important!!!\n         * Sanitizing HTML before displaying on webpage to prevent XSS attacks!!\n         */\n        html = DOMPurify.sanitize(md.render(html));\n        this._element.innerHTML = html;\n        log.debug(this._element.innerHTML);\n    }\n}","import { Observer } from \"../observe/Observer\";\nimport { Model } from \"../model/AbstractModel\";\nimport { Subject } from \"../observe/Observable\";\nimport { View } from \"./AbstractView\";\nimport { Controller } from \"../controller/AbstractController\";\nimport { TemplateFactory } from \"../template/TemplateFactory\";\nimport { ActiveUserViewModel } from \"../viewmodel/ActiveUserViewModel\";\nimport { ChatModel } from \"../model/ChatModel\";\nimport log = require(\"loglevel\");\n\nexport class UserView implements Observer {\n    private readonly _model: Model;\n    private readonly _chatModel: ChatModel;\n    private readonly _element: HTMLElement;\n    // private userBoxes: any[] =  [];\n\n\n    constructor(model: Model, chatModel: ChatModel, element: HTMLElement) {\n        this._model = model;\n        this._chatModel = chatModel;\n        this._element = element;\n    }\n\n\n\n\n    update(data: ActiveUserViewModel[]): void {\n        let template = TemplateFactory.getTemplate('user-contact-online-template');\n        let html: string = \"\";\n        data.forEach((element: ActiveUserViewModel) => {\n            html += template(element);\n        });\n        this._element.innerHTML = html;\n        this.addUserCallBacks();\n        console.log(this._element.innerHTML);\n    }\n\n    private helper(): void {\n\n    }\n\n    private addUserCallBacks(): void {\n        let userBoxes = document.getElementsByClassName('user-box');\n        for (let i = 0; i < userBoxes.length; i++) {\n            let userBox = userBoxes[i];\n            userBoxes[i].addEventListener('click', this.userCallBack.bind(this, userBox));\n        }\n    }\n\n\n    private userCallBack(el: Element): void {\n        let current = document.getElementsByClassName('user-box active');\n\n        let passphrase: string = '';\n        if (current.length > 0) {\n            let passphraseInput = document.getElementById('passphrase') as any;\n\n            if (passphraseInput == null) {\n                log.error('passphraseInput element reference is null');\n                return;\n            }\n            passphrase = passphraseInput.value\n            if (passphrase == '' || passphrase == null) {\n                // alert('Please input passphrase')\n                // alertify.error('Please enter a passphrase');\n                log.error('passphrase is empty or null');\n                return;\n            }\n            current[0].className = current[0].className.replace(\" active\", \"\");\n\n        }\n        // Add the active class to the current/clicked button\n        else if (current.length == 0) {\n            let elem = document.getElementById('passphrase-initial') as any;\n            if(elem == null)\n            {\n                log.error('passphraseInput element reference is null');\n                return;\n            }\n            passphrase = elem.value;\n            if (passphrase == '' || passphrase == null) {\n                //     // alert('Please input passphrase')\n                //     // alertify.error('Please enter a passphrase');\n                log.error('passphrase is empty or null');\n                return;\n            }\n            // @ts-ignore: Object is possibly 'null'.\n            document.getElementById('no-user-selected').hidden = true;\n            // @ts-ignore: Object is possibly 'null'.\n            document.getElementById('chat-card').hidden = false;\n            // @ts-ignore: Object is possibly 'null'.\n            elem.hidden = true;\n        }\n        // console.log(this.getElementsByClassName('to-user-span'));\n        let elem = el.getElementsByClassName('to-user-span')[0] as HTMLElement;\n        let userName = elem.innerText;\n        // @ts-ignore: Object is possibly 'null'.\n        document.getElementById('user-name-span').innerText = userName;\n        this._chatModel.getmessages(userName, passphrase, null);\n        // populateMessages(userName, passphrase);\n        sessionStorage.setItem('selectedUser', userName);\n        el.className += \" active\";\n    }\n\n}","export class ActiveUserViewModel {\n    userName: string | undefined;\n    online: boolean | undefined;\n    lastActive: string| undefined;\n}","export class ChatMessageViewModel {\n    public toUser: string | undefined;\n    public fromUser: string | undefined;\n    public message!: string;\n    public messageTime!: Date;\n\n    \n}"]} 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 @@ -
+