forked from cory/tildefriends
		
	Work in progress modular, resumable, multi-protocol chat client.
git-svn-id: https://www.unprompted.com/svn/projects/tildefriends/trunk@3228 ed5197a5-7fde-0310-b194-c3ffbd925b24
This commit is contained in:
		
							
								
								
									
										379
									
								
								packages/cory/chat/chat.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										379
									
								
								packages/cory/chat/chat.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,379 @@ | ||||
| "use strict"; | ||||
|  | ||||
| var gFocus = true; | ||||
| var gUnread = 0; | ||||
| var gPresence = {}; | ||||
| let gSessions = {}; | ||||
| let gCurrentConversation; | ||||
|  | ||||
| function updateTitle() { | ||||
| 	terminal.setTitle((gUnread ? "(" + gUnread.toString() + ") " : "") + "Chat"); | ||||
| } | ||||
|  | ||||
| let kAccountsKey = JSON.stringify(["accounts", core.user.name]); | ||||
|  | ||||
| function runCommand(data) { | ||||
| 	if (data.action == "addAccount") { | ||||
| 		addAccount(); | ||||
| 	} else if (data.action == "deleteAccount") { | ||||
| 		deleteAccount(data.id); | ||||
| 	} else if (data.action == "updateAccount") { | ||||
| 		delete data.action; | ||||
| 		let id = data.id; | ||||
| 		delete data.id; | ||||
| 		configureAccount(id, data); | ||||
| 	} else if (data.action == "connect") { | ||||
| 		connect(data.id); | ||||
| 	} else if (data.action == "disconnect") { | ||||
| 		disconnect(data.id); | ||||
| 	} else if (data.action == "window") { | ||||
| 		gCurrentConversation = gSessions[data.account].conversations[data.conversation]; | ||||
| 		updateConversation(); | ||||
| 		updateWindows(); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| function addAccount() { | ||||
| 	return database.get(kAccountsKey).then(function(data) { | ||||
| 		let accounts = data ? JSON.parse(data) : []; | ||||
| 		let id = 0; | ||||
| 		for (var i in accounts) { | ||||
| 			id = Math.max(id, accounts[i].id + 1); | ||||
| 		} | ||||
| 		accounts.push({name: "unnamed", id: id}); | ||||
| 		return database.set(kAccountsKey, JSON.stringify(accounts)); | ||||
| 	}).then(updateWindows); | ||||
| } | ||||
|  | ||||
| core.register("submit", function(data) { | ||||
| 	if (data.value.submit == "Save Account") { | ||||
| 		let id = data.value.id; | ||||
| 		delete data.value.id; | ||||
| 		delete data.value.submit; | ||||
| 		configureAccount(id, data.value); | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| function configureAccount(id, updates) { | ||||
| 	return Promise.all([database.get(kAccountsKey), core.getPackages()]).then(function(results) { | ||||
| 		let accounts = results[0] ? JSON.parse(results[0]) : []; | ||||
| 		let packages = results[1]; | ||||
| 		let account; | ||||
| 		let accountIndex; | ||||
| 		for (let i in accounts) { | ||||
| 			if (accounts[i].id == id) { | ||||
| 				account = accounts[i]; | ||||
| 				accountIndex = i; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		let promises = []; | ||||
|  | ||||
| 		if (updates) { | ||||
| 			for (let i in updates) { | ||||
| 				account[i] = updates[i]; | ||||
| 			} | ||||
| 			promises.push(database.set(kAccountsKey, JSON.stringify(accounts))); | ||||
| 		} | ||||
|  | ||||
| 		return Promise.all(promises).then(function() { | ||||
| 			terminal.clear(); | ||||
| 			terminal.print(JSON.stringify(account)); | ||||
| 			terminal.print({input: "hidden", value: id, name: "id"}); | ||||
| 			terminal.print({input: "text", value: account.name, name: "name"}); | ||||
| 			terminal.print({command: "/command " + JSON.stringify({action: "deleteAccount", id: id}), value: "delete account"}); | ||||
| 			terminal.print({command: "/command " + JSON.stringify({action: "connect", id: id}), value: "connect"}); | ||||
| 			terminal.print({command: "/command " + JSON.stringify({action: "disconnect", id: id}), value: "disconnect"}); | ||||
|  | ||||
| 			if (!account.type) { | ||||
| 				terminal.print("Pick account type:"); | ||||
| 				for (let i in packages) { | ||||
| 					let app = packages[i]; | ||||
| 					if (app.manifest && app.manifest.chat) { | ||||
| 						terminal.print({command: "/command " + JSON.stringify({action: "updateAccount", id: id, type: app.name}), value: app.name}); | ||||
| 					} | ||||
| 				} | ||||
| 			} else { | ||||
| 				let schema; | ||||
| 				for (let i in packages) { | ||||
| 					let app = packages[i]; | ||||
| 					if (app.name == account.type && app.manifest && app.manifest.chat) { | ||||
| 						schema = app.manifest.chat.settings; | ||||
| 						break; | ||||
| 					} | ||||
| 				} | ||||
| 				if (schema) { | ||||
| 					for (var i in schema) { | ||||
| 						let field = schema[i]; | ||||
| 						terminal.print({input: field.type, name: field.name, value: account[field.name] || field.default}); | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			terminal.print({input: "submit", value: "Save Account", name: "submit"}); | ||||
| 		}).then(updateWindows); | ||||
| 	}).catch(function(error) { | ||||
| 		print("whoops", error); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| function deleteAccount(id) { | ||||
| 	return database.get(kAccountsKey).then(function(data) { | ||||
| 		let accounts = data ? JSON.parse(data) : []; | ||||
| 		for (var i = 0; i < accounts.length; i++) { | ||||
| 			if (accounts[i] && (!accounts[i].id || accounts[i].id == id)) { | ||||
| 				accounts.splice(i, 1); | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
| 		return database.set(kAccountsKey, JSON.stringify(accounts)); | ||||
| 	}).then(terminal.clear).then(updateWindows); | ||||
| } | ||||
|  | ||||
| function connect(id) { | ||||
| 	return database.get(kAccountsKey).then(function(data) { | ||||
| 		let accounts = data ? JSON.parse(data) : []; | ||||
| 		let account; | ||||
| 		for (var i = 0; i < accounts.length; i++) { | ||||
| 			if (accounts[i] && (!accounts[i].id || accounts[i].id == id)) { | ||||
| 				account = accounts[i]; | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (account) { | ||||
| 			let self = {account: account}; | ||||
| 			let options = {callback: chatCallback.bind(self)}; | ||||
| 			for (var i in account) { | ||||
| 				options[i] = account[i]; | ||||
| 			} | ||||
| 			return core.getService("chat", account.type).then(function(service) { | ||||
| 				return service.postMessage(options).then(function(sessions) { | ||||
| 					let session = sessions[0]; | ||||
| 					self.session = session; | ||||
| 					gSessions[id] = session; | ||||
| 					session.conversations = {}; | ||||
| 					getConversation(session, {}); | ||||
| 					session.getConversations().then(function(conversations) { | ||||
| 						print(conversations); | ||||
| 						for (let j in conversations) { | ||||
| 							getConversation(session, {conversation: conversations[j]}); | ||||
| 						} | ||||
| 					}); | ||||
| 				}); | ||||
| 			}); | ||||
| 		} | ||||
| 	}).catch(function(error) { | ||||
| 		print(error); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| function disconnect(id) { | ||||
| 	gSessions[id].disconnect(); | ||||
| } | ||||
|  | ||||
| function updateWindows() { | ||||
| 	database.get(kAccountsKey).then(function(data) { | ||||
| 		let accounts = data ? JSON.parse(data) : []; | ||||
|  | ||||
| 		terminal.cork(); | ||||
| 		terminal.select("windows"); | ||||
| 		terminal.clear(); | ||||
| 		terminal.print({style: "font-size: x-large", value: "Windows"}); | ||||
| 		for (let i in accounts) { | ||||
| 			let account = accounts[i]; | ||||
| 			terminal.print({style: "font-size: large", command: "/command " + JSON.stringify({action: "updateAccount", id: account.id}), value: account.name}); | ||||
| 			if (gSessions[account.id] && gSessions[account.id].conversations) { | ||||
| 				let conversations = gSessions[account.id].conversations; | ||||
| 				for (let j in conversations) { | ||||
| 					terminal.print({ | ||||
| 						command: "/command " + JSON.stringify({action: "window", account: account.id, conversation: j}), | ||||
| 						value: j ? j : "<service>", | ||||
| 						style: (conversations[j] == gCurrentConversation ? "font-weight: bold; " : "") + "color: white", | ||||
| 					}); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		terminal.print({style: "color: yellow", command: "/command " + JSON.stringify({action: "addAccount"}), value: "add account"}); | ||||
| 		terminal.select("terminal"); | ||||
| 		terminal.uncork(); | ||||
| 	}).catch(function(error) { | ||||
| 		print(error); | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| function updateConversation() { | ||||
| 	if (gCurrentConversation) { | ||||
| 		Promise.all([ | ||||
| 			gCurrentConversation.session.getHistory(gCurrentConversation.name), | ||||
| 			gCurrentConversation.session.getParticipants(gCurrentConversation.name), | ||||
| 		]).then(function(data) { | ||||
| 			print(data); | ||||
| 			let history = data[0]; | ||||
| 			let participants = data[1]; | ||||
| 			gCurrentConversation.messages = history; | ||||
| 			gCurrentConversation.participants = participants; | ||||
| 			terminal.cork(); | ||||
| 			terminal.select("terminal"); | ||||
| 			terminal.clear(); | ||||
| 			for (var i in gCurrentConversation.messages) { | ||||
| 				printMessage(gCurrentConversation.messages[i]); | ||||
| 			} | ||||
| 			updateUsers(); | ||||
| 			terminal.uncork(); | ||||
| 		}).catch(function(error) { | ||||
| 			terminal.print(error); | ||||
| 		}); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| function updateUsers() { | ||||
| 	terminal.cork(); | ||||
| 	terminal.select("users"); | ||||
| 	terminal.clear(); | ||||
| 	terminal.print({style: "font-size: x-large", value: "Users"}); | ||||
| 	if (gCurrentConversation) { | ||||
| 		for (var i in gCurrentConversation.participants) { | ||||
| 			terminal.print(gCurrentConversation.participants[i]); | ||||
| 		} | ||||
| 	} | ||||
| 	terminal.select("terminal"); | ||||
| 	terminal.uncork(); | ||||
| } | ||||
|  | ||||
| terminal.cork(); | ||||
| terminal.split([ | ||||
| 	{type: "horizontal", children: [ | ||||
| 		{name: "windows", basis: "2in", grow: "0", shrink: "0"}, | ||||
| 		{name: "terminal", grow: "1"}, | ||||
| 		{name: "users", basis: "2in", grow: "0", shrink: "0"}, | ||||
| 	]}, | ||||
| ]); | ||||
| updateTitle(); | ||||
| updateWindows(); | ||||
| updateUsers(); | ||||
| terminal.setEcho(false); | ||||
| terminal.select("terminal"); | ||||
| terminal.print("~Friends Chat"); | ||||
| terminal.uncork(); | ||||
|  | ||||
| function getConversation(session, message) { | ||||
| 	let result; | ||||
| 	for (var i in gSessions) { | ||||
| 		if (session == gSessions[i]) { | ||||
| 			let key = message.conversation || message.from || ""; | ||||
| 			if (!session.conversations[key]) { | ||||
| 				session.conversations[key] = { | ||||
| 					session: session, | ||||
| 					name: key, | ||||
| 					messages: [], | ||||
| 					sendMessage: function(message) { | ||||
| 						return session.sendMessage(key, message); | ||||
| 					}, | ||||
| 				}; | ||||
| 				updateWindows(); | ||||
| 			} | ||||
| 			result = session.conversations[key]; | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
| 	if (result && !gCurrentConversation) { | ||||
| 		gCurrentConversation = result; | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| function chatCallback(event) { | ||||
| 	print(event); | ||||
| 	if (event.action == "message") { | ||||
| 		let conversation = getConversation(this.session, event); | ||||
| 		if (conversation == gCurrentConversation) { | ||||
| 			printMessage(event); | ||||
| 		} | ||||
| 		conversation.messages.push(event); | ||||
|  | ||||
| 		if (!gFocus) { | ||||
| 			gUnread++; | ||||
| 			updateTitle(); | ||||
| 		} | ||||
| 	} else { | ||||
| 		terminal.print("Unhandled event: ", JSON.stringify(event)); | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| core.register("onInput", function(input) { | ||||
| 	if (input.substring(0, "/command ".length) == "/command ") { | ||||
| 		runCommand(JSON.parse(input.substring("/command ".length))); | ||||
| 	} else if (gCurrentConversation) { | ||||
| 		gCurrentConversation.sendMessage(input).catch(function(error) { | ||||
| 			terminal.print("Message not sent: ", error); | ||||
| 		}); | ||||
| 	} | ||||
| }); | ||||
|  | ||||
| function niceTime(lastTime, thisTime) { | ||||
| 	if (!lastTime) { | ||||
| 		return thisTime; | ||||
| 	} | ||||
| 	let result = []; | ||||
| 	let lastParts = lastTime.split(" "); | ||||
| 	let thisParts = thisTime.split(" "); | ||||
| 	for (let i = 0; i < thisParts.length; i++) { | ||||
| 		if (thisParts[i] !== lastParts[i]) { | ||||
| 			result.push(thisParts[i]); | ||||
| 		} | ||||
| 	} | ||||
| 	return result.join(" "); | ||||
| } | ||||
|  | ||||
| function formatMessage(message) { | ||||
| 	var result; | ||||
| 	if (typeof message == "string") { | ||||
| 		result = []; | ||||
| 		var regex = /(\w+:\/*\S+?)(?=(?:[\.!?])?(?:$|\s))/gi; | ||||
| 		var match; | ||||
| 		var lastIndex = 0; | ||||
| 		while ((match = regex.exec(message)) !== null) { | ||||
| 			result.push({class: "base1", value: message.substring(lastIndex, match.index)}); | ||||
| 			result.push({href: match[0]}); | ||||
| 			lastIndex = regex.lastIndex; | ||||
| 		} | ||||
| 		result.push({class: "base1", value: message.substring(lastIndex)}); | ||||
| 	} else { | ||||
| 		result = message; | ||||
| 	} | ||||
| 	return result; | ||||
| } | ||||
|  | ||||
| var lastTimestamp = null; | ||||
| function printMessage(message) { | ||||
| 	var now = message.timestamp || new Date().toString(); | ||||
| 	var from = message.from || "unknown"; | ||||
|  | ||||
| 	terminal.print( | ||||
| 		{class: "base0", value: niceTime(lastTimestamp, now)}, | ||||
| 		" ", | ||||
| 		{class: "base00", value: "<"}, | ||||
| 		{class: "base3", value: from}, | ||||
| 		{class: "base00", value: ">"}, | ||||
| 		" ", | ||||
| 		formatMessage(message.message)); | ||||
| 	lastTimestamp = now; | ||||
| } | ||||
|  | ||||
| core.register("focus", function() { | ||||
| 	gFocus = true; | ||||
| 	gUnread = 0; | ||||
| 	updateTitle(); | ||||
| }); | ||||
|  | ||||
| core.register("blur", function() { | ||||
| 	gFocus = false; | ||||
| }); | ||||
|  | ||||
| // Connect all accounts on start. | ||||
| Promise.all([database.get(kAccountsKey), core.getPackages()]).then(function(results) { | ||||
| 	let accounts = results[0] ? JSON.parse(results[0]) : []; | ||||
| 	for (let i in accounts) { | ||||
| 		connect(accounts[i].id); | ||||
| 	} | ||||
| }); | ||||
							
								
								
									
										941
									
								
								packages/cory/xmpp2/xmpp2.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										941
									
								
								packages/cory/xmpp2/xmpp2.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,941 @@ | ||||
| "use strict"; | ||||
|  | ||||
| //! { | ||||
| //! 	"permissions": [ | ||||
| //! 		"network" | ||||
| //! 	], | ||||
| //! 	"chat": { | ||||
| //! 		"version": 1, | ||||
| //! 		"settings": [ | ||||
| //! 			{"name": "userName", "type": "text"}, | ||||
| //! 			{"name": "password", "type": "password"}, | ||||
| //! 			{"name": "resource", "type": "text", "default": "tildefriends"}, | ||||
| //! 			{"name": "server", "type": "text"} | ||||
| //! 		] | ||||
| //! 	} | ||||
| //! } | ||||
|  | ||||
| // md5.js | ||||
|  | ||||
| /* | ||||
|  * JavaScript MD5 1.0.1 | ||||
|  * https://github.com/blueimp/JavaScript-MD5 | ||||
|  * | ||||
|  * Copyright 2011, Sebastian Tschan | ||||
|  * https://blueimp.net | ||||
|  * | ||||
|  * Licensed under the MIT license: | ||||
|  * http://www.opensource.org/licenses/MIT | ||||
|  * | ||||
|  * Based on | ||||
|  * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message | ||||
|  * Digest Algorithm, as defined in RFC 1321. | ||||
|  * Version 2.2 Copyright (C) Paul Johnston 1999 - 2009 | ||||
|  * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet | ||||
|  * Distributed under the BSD License | ||||
|  * See http://pajhome.org.uk/crypt/md5 for more info. | ||||
|  */ | ||||
|  | ||||
| /*jslint bitwise: true */ | ||||
| /*global unescape, define */ | ||||
|  | ||||
| 'use strict'; | ||||
|  | ||||
| /* | ||||
|  * Add integers, wrapping at 2^32. This uses 16-bit operations internally | ||||
|  * to work around bugs in some JS interpreters. | ||||
|  */ | ||||
| function safe_add(x, y) { | ||||
| 	var lsw = (x & 0xFFFF) + (y & 0xFFFF), | ||||
| 		msw = (x >> 16) + (y >> 16) + (lsw >> 16); | ||||
| 	return (msw << 16) | (lsw & 0xFFFF); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Bitwise rotate a 32-bit number to the left. | ||||
|  */ | ||||
| function bit_rol(num, cnt) { | ||||
| 	return (num << cnt) | (num >>> (32 - cnt)); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * These functions implement the four basic operations the algorithm uses. | ||||
|  */ | ||||
| function md5_cmn(q, a, b, x, s, t) { | ||||
| 	return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s), b); | ||||
| } | ||||
| function md5_ff(a, b, c, d, x, s, t) { | ||||
| 	return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t); | ||||
| } | ||||
| function md5_gg(a, b, c, d, x, s, t) { | ||||
| 	return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t); | ||||
| } | ||||
| function md5_hh(a, b, c, d, x, s, t) { | ||||
| 	return md5_cmn(b ^ c ^ d, a, b, x, s, t); | ||||
| } | ||||
| function md5_ii(a, b, c, d, x, s, t) { | ||||
| 	return md5_cmn(c ^ (b | (~d)), a, b, x, s, t); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Calculate the MD5 of an array of little-endian words, and a bit length. | ||||
|  */ | ||||
| function binl_md5(x, len) { | ||||
| 	/* append padding */ | ||||
| 	x[len >> 5] |= 0x80 << (len % 32); | ||||
| 	x[(((len + 64) >>> 9) << 4) + 14] = len; | ||||
|  | ||||
| 	var i, olda, oldb, oldc, oldd, | ||||
| 		a =  1732584193, | ||||
| 		b = -271733879, | ||||
| 		c = -1732584194, | ||||
| 		d =  271733878; | ||||
|  | ||||
| 	for (i = 0; i < x.length; i += 16) { | ||||
| 		olda = a; | ||||
| 		oldb = b; | ||||
| 		oldc = c; | ||||
| 		oldd = d; | ||||
|  | ||||
| 		a = md5_ff(a, b, c, d, x[i],       7, -680876936); | ||||
| 		d = md5_ff(d, a, b, c, x[i +  1], 12, -389564586); | ||||
| 		c = md5_ff(c, d, a, b, x[i +  2], 17,  606105819); | ||||
| 		b = md5_ff(b, c, d, a, x[i +  3], 22, -1044525330); | ||||
| 		a = md5_ff(a, b, c, d, x[i +  4],  7, -176418897); | ||||
| 		d = md5_ff(d, a, b, c, x[i +  5], 12,  1200080426); | ||||
| 		c = md5_ff(c, d, a, b, x[i +  6], 17, -1473231341); | ||||
| 		b = md5_ff(b, c, d, a, x[i +  7], 22, -45705983); | ||||
| 		a = md5_ff(a, b, c, d, x[i +  8],  7,  1770035416); | ||||
| 		d = md5_ff(d, a, b, c, x[i +  9], 12, -1958414417); | ||||
| 		c = md5_ff(c, d, a, b, x[i + 10], 17, -42063); | ||||
| 		b = md5_ff(b, c, d, a, x[i + 11], 22, -1990404162); | ||||
| 		a = md5_ff(a, b, c, d, x[i + 12],  7,  1804603682); | ||||
| 		d = md5_ff(d, a, b, c, x[i + 13], 12, -40341101); | ||||
| 		c = md5_ff(c, d, a, b, x[i + 14], 17, -1502002290); | ||||
| 		b = md5_ff(b, c, d, a, x[i + 15], 22,  1236535329); | ||||
|  | ||||
| 		a = md5_gg(a, b, c, d, x[i +  1],  5, -165796510); | ||||
| 		d = md5_gg(d, a, b, c, x[i +  6],  9, -1069501632); | ||||
| 		c = md5_gg(c, d, a, b, x[i + 11], 14,  643717713); | ||||
| 		b = md5_gg(b, c, d, a, x[i],      20, -373897302); | ||||
| 		a = md5_gg(a, b, c, d, x[i +  5],  5, -701558691); | ||||
| 		d = md5_gg(d, a, b, c, x[i + 10],  9,  38016083); | ||||
| 		c = md5_gg(c, d, a, b, x[i + 15], 14, -660478335); | ||||
| 		b = md5_gg(b, c, d, a, x[i +  4], 20, -405537848); | ||||
| 		a = md5_gg(a, b, c, d, x[i +  9],  5,  568446438); | ||||
| 		d = md5_gg(d, a, b, c, x[i + 14],  9, -1019803690); | ||||
| 		c = md5_gg(c, d, a, b, x[i +  3], 14, -187363961); | ||||
| 		b = md5_gg(b, c, d, a, x[i +  8], 20,  1163531501); | ||||
| 		a = md5_gg(a, b, c, d, x[i + 13],  5, -1444681467); | ||||
| 		d = md5_gg(d, a, b, c, x[i +  2],  9, -51403784); | ||||
| 		c = md5_gg(c, d, a, b, x[i +  7], 14,  1735328473); | ||||
| 		b = md5_gg(b, c, d, a, x[i + 12], 20, -1926607734); | ||||
|  | ||||
| 		a = md5_hh(a, b, c, d, x[i +  5],  4, -378558); | ||||
| 		d = md5_hh(d, a, b, c, x[i +  8], 11, -2022574463); | ||||
| 		c = md5_hh(c, d, a, b, x[i + 11], 16,  1839030562); | ||||
| 		b = md5_hh(b, c, d, a, x[i + 14], 23, -35309556); | ||||
| 		a = md5_hh(a, b, c, d, x[i +  1],  4, -1530992060); | ||||
| 		d = md5_hh(d, a, b, c, x[i +  4], 11,  1272893353); | ||||
| 		c = md5_hh(c, d, a, b, x[i +  7], 16, -155497632); | ||||
| 		b = md5_hh(b, c, d, a, x[i + 10], 23, -1094730640); | ||||
| 		a = md5_hh(a, b, c, d, x[i + 13],  4,  681279174); | ||||
| 		d = md5_hh(d, a, b, c, x[i],      11, -358537222); | ||||
| 		c = md5_hh(c, d, a, b, x[i +  3], 16, -722521979); | ||||
| 		b = md5_hh(b, c, d, a, x[i +  6], 23,  76029189); | ||||
| 		a = md5_hh(a, b, c, d, x[i +  9],  4, -640364487); | ||||
| 		d = md5_hh(d, a, b, c, x[i + 12], 11, -421815835); | ||||
| 		c = md5_hh(c, d, a, b, x[i + 15], 16,  530742520); | ||||
| 		b = md5_hh(b, c, d, a, x[i +  2], 23, -995338651); | ||||
|  | ||||
| 		a = md5_ii(a, b, c, d, x[i],       6, -198630844); | ||||
| 		d = md5_ii(d, a, b, c, x[i +  7], 10,  1126891415); | ||||
| 		c = md5_ii(c, d, a, b, x[i + 14], 15, -1416354905); | ||||
| 		b = md5_ii(b, c, d, a, x[i +  5], 21, -57434055); | ||||
| 		a = md5_ii(a, b, c, d, x[i + 12],  6,  1700485571); | ||||
| 		d = md5_ii(d, a, b, c, x[i +  3], 10, -1894986606); | ||||
| 		c = md5_ii(c, d, a, b, x[i + 10], 15, -1051523); | ||||
| 		b = md5_ii(b, c, d, a, x[i +  1], 21, -2054922799); | ||||
| 		a = md5_ii(a, b, c, d, x[i +  8],  6,  1873313359); | ||||
| 		d = md5_ii(d, a, b, c, x[i + 15], 10, -30611744); | ||||
| 		c = md5_ii(c, d, a, b, x[i +  6], 15, -1560198380); | ||||
| 		b = md5_ii(b, c, d, a, x[i + 13], 21,  1309151649); | ||||
| 		a = md5_ii(a, b, c, d, x[i +  4],  6, -145523070); | ||||
| 		d = md5_ii(d, a, b, c, x[i + 11], 10, -1120210379); | ||||
| 		c = md5_ii(c, d, a, b, x[i +  2], 15,  718787259); | ||||
| 		b = md5_ii(b, c, d, a, x[i +  9], 21, -343485551); | ||||
|  | ||||
| 		a = safe_add(a, olda); | ||||
| 		b = safe_add(b, oldb); | ||||
| 		c = safe_add(c, oldc); | ||||
| 		d = safe_add(d, oldd); | ||||
| 	} | ||||
| 	return [a, b, c, d]; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Convert an array of little-endian words to a string | ||||
|  */ | ||||
| function binl2rstr(input) { | ||||
| 	var i, | ||||
| 		output = ''; | ||||
| 	for (i = 0; i < input.length * 32; i += 8) { | ||||
| 		output += String.fromCharCode((input[i >> 5] >>> (i % 32)) & 0xFF); | ||||
| 	} | ||||
| 	return output; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Convert a raw string to an array of little-endian words | ||||
|  * Characters >255 have their high-byte silently ignored. | ||||
|  */ | ||||
| function rstr2binl(input) { | ||||
| 	var i, | ||||
| 		output = []; | ||||
| 	output[(input.length >> 2) - 1] = undefined; | ||||
| 	for (i = 0; i < output.length; i += 1) { | ||||
| 		output[i] = 0; | ||||
| 	} | ||||
| 	for (i = 0; i < input.length * 8; i += 8) { | ||||
| 		output[i >> 5] |= (input.charCodeAt(i / 8) & 0xFF) << (i % 32); | ||||
| 	} | ||||
| 	return output; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Calculate the MD5 of a raw string | ||||
|  */ | ||||
| function rstr_md5(s) { | ||||
| 	return binl2rstr(binl_md5(rstr2binl(s), s.length * 8)); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Calculate the HMAC-MD5, of a key and some data (raw strings) | ||||
|  */ | ||||
| function rstr_hmac_md5(key, data) { | ||||
| 	var i, | ||||
| 		bkey = rstr2binl(key), | ||||
| 		ipad = [], | ||||
| 		opad = [], | ||||
| 		hash; | ||||
| 	ipad[15] = opad[15] = undefined; | ||||
| 	if (bkey.length > 16) { | ||||
| 		bkey = binl_md5(bkey, key.length * 8); | ||||
| 	} | ||||
| 	for (i = 0; i < 16; i += 1) { | ||||
| 		ipad[i] = bkey[i] ^ 0x36363636; | ||||
| 		opad[i] = bkey[i] ^ 0x5C5C5C5C; | ||||
| 	} | ||||
| 	hash = binl_md5(ipad.concat(rstr2binl(data)), 512 + data.length * 8); | ||||
| 	return binl2rstr(binl_md5(opad.concat(hash), 512 + 128)); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Convert a raw string to a hex string | ||||
|  */ | ||||
| function rstr2hex(input) { | ||||
| 	var hex_tab = '0123456789abcdef', | ||||
| 		output = '', | ||||
| 		x, | ||||
| 		i; | ||||
| 	for (i = 0; i < input.length; i += 1) { | ||||
| 		x = input.charCodeAt(i); | ||||
| 		output += hex_tab.charAt((x >>> 4) & 0x0F) + | ||||
| 			hex_tab.charAt(x & 0x0F); | ||||
| 	} | ||||
| 	return output; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Encode a string as utf-8 | ||||
|  */ | ||||
| function str2rstr_utf8(input) { | ||||
| 	return unescape(input); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Take string arguments and return either raw or hex encoded strings | ||||
|  */ | ||||
| function raw_md5(s) { | ||||
| 	return rstr_md5(str2rstr_utf8(s)); | ||||
| } | ||||
| function hex_md5(s) { | ||||
| 	return rstr2hex(raw_md5(s)); | ||||
| } | ||||
| function raw_hmac_md5(k, d) { | ||||
| 	return rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d)); | ||||
| } | ||||
| function hex_hmac_md5(k, d) { | ||||
| 	return rstr2hex(raw_hmac_md5(k, d)); | ||||
| } | ||||
|  | ||||
| function md5(string, key, raw) { | ||||
| 	if (!key) { | ||||
| 		if (!raw) { | ||||
| 			return hex_md5(string); | ||||
| 		} | ||||
| 		return raw_md5(string); | ||||
| 	} | ||||
| 	if (!raw) { | ||||
| 		return hex_hmac_md5(key, string); | ||||
| 	} | ||||
| 	return raw_hmac_md5(key, string); | ||||
| } | ||||
|  | ||||
| // end md5.js | ||||
|  | ||||
| // base64.js | ||||
| /** | ||||
| * | ||||
| *  Base64 encode / decode | ||||
| *  http://www.webtoolkit.info/ | ||||
| * | ||||
| **/ | ||||
|  | ||||
| // private property | ||||
| var _keyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/="; | ||||
|  | ||||
| var Base64 = { | ||||
|  | ||||
| // public method for encoding | ||||
| encode : function (input) { | ||||
|     var output = ""; | ||||
|     var chr1, chr2, chr3, enc1, enc2, enc3, enc4; | ||||
|     var i = 0; | ||||
|  | ||||
|     input = Base64._utf8_encode(input); | ||||
|  | ||||
|     while (i < input.length) { | ||||
|  | ||||
|         chr1 = input.charCodeAt(i++); | ||||
|         chr2 = input.charCodeAt(i++); | ||||
|         chr3 = input.charCodeAt(i++); | ||||
|  | ||||
|         enc1 = chr1 >> 2; | ||||
|         enc2 = ((chr1 & 3) << 4) | (chr2 >> 4); | ||||
|         enc3 = ((chr2 & 15) << 2) | (chr3 >> 6); | ||||
|         enc4 = chr3 & 63; | ||||
|  | ||||
|         if (isNaN(chr2)) { | ||||
|             enc3 = enc4 = 64; | ||||
|         } else if (isNaN(chr3)) { | ||||
|             enc4 = 64; | ||||
|         } | ||||
|  | ||||
|         output = output + | ||||
|         _keyStr.charAt(enc1) + _keyStr.charAt(enc2) + | ||||
|         _keyStr.charAt(enc3) + _keyStr.charAt(enc4); | ||||
|  | ||||
|     } | ||||
|  | ||||
|     return output; | ||||
| }, | ||||
|  | ||||
| // public method for decoding | ||||
| decode : function (input) { | ||||
|     var output = ""; | ||||
|     var chr1, chr2, chr3; | ||||
|     var enc1, enc2, enc3, enc4; | ||||
|     var i = 0; | ||||
|  | ||||
|     input = input.replace(/[^A-Za-z0-9\+\/\=]/g, ""); | ||||
|  | ||||
|     while (i < input.length) { | ||||
|  | ||||
|         enc1 = _keyStr.indexOf(input.charAt(i++)); | ||||
|         enc2 = _keyStr.indexOf(input.charAt(i++)); | ||||
|         enc3 = _keyStr.indexOf(input.charAt(i++)); | ||||
|         enc4 = _keyStr.indexOf(input.charAt(i++)); | ||||
|  | ||||
|         chr1 = (enc1 << 2) | (enc2 >> 4); | ||||
|         chr2 = ((enc2 & 15) << 4) | (enc3 >> 2); | ||||
|         chr3 = ((enc3 & 3) << 6) | enc4; | ||||
|  | ||||
|         output = output + String.fromCharCode(chr1); | ||||
|  | ||||
|         if (enc3 != 64) { | ||||
|             output = output + String.fromCharCode(chr2); | ||||
|         } | ||||
|         if (enc4 != 64) { | ||||
|             output = output + String.fromCharCode(chr3); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     output = Base64._utf8_decode(output); | ||||
|  | ||||
|     return output; | ||||
|  | ||||
| }, | ||||
|  | ||||
| // private method for UTF-8 encoding | ||||
| _utf8_encode : function (string) { | ||||
|     string = string.replace(/\r\n/g,"\n"); | ||||
|     var utftext = ""; | ||||
|  | ||||
|     for (var n = 0; n < string.length; n++) { | ||||
|  | ||||
|         var c = string.charCodeAt(n); | ||||
|  | ||||
|         if (c < 128) { | ||||
|             utftext += String.fromCharCode(c); | ||||
|         } | ||||
|         else if((c > 127) && (c < 2048)) { | ||||
|             utftext += String.fromCharCode((c >> 6) | 192); | ||||
|             utftext += String.fromCharCode((c & 63) | 128); | ||||
|         } | ||||
|         else { | ||||
|             utftext += String.fromCharCode((c >> 12) | 224); | ||||
|             utftext += String.fromCharCode(((c >> 6) & 63) | 128); | ||||
|             utftext += String.fromCharCode((c & 63) | 128); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     return utftext; | ||||
| }, | ||||
|  | ||||
| // private method for UTF-8 decoding | ||||
| _utf8_decode : function (utftext) { | ||||
|     var string = ""; | ||||
|     var i = 0; | ||||
|     var c = 0; | ||||
|     var c1 = 0; | ||||
|     var c2 = 0; | ||||
|  | ||||
|     while ( i < utftext.length ) { | ||||
|  | ||||
|         c = utftext.charCodeAt(i); | ||||
|  | ||||
|         if (c < 128) { | ||||
|             string += String.fromCharCode(c); | ||||
|             i++; | ||||
|         } | ||||
|         else if((c > 191) && (c < 224)) { | ||||
|             c2 = utftext.charCodeAt(i+1); | ||||
|             string += String.fromCharCode(((c & 31) << 6) | (c2 & 63)); | ||||
|             i += 2; | ||||
|         } | ||||
|         else { | ||||
|             c2 = utftext.charCodeAt(i+1); | ||||
|             c3 = utftext.charCodeAt(i+2); | ||||
|             string += String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)); | ||||
|             i += 3; | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     return string; | ||||
| } | ||||
|  | ||||
| } | ||||
|  | ||||
| // end base64.js | ||||
|  | ||||
| function xmlEncode(text) { | ||||
| 	return text.replace(/([\&"'<>])/g, function(x, item) { | ||||
| 		return {'&': '&', '"': '"', '<': '<', '>': '>', "'": '''}[item]; | ||||
| 	}); | ||||
| } | ||||
| function xmlDecode(xml) { | ||||
| 	return xml.replace(/("|<|>|&|')/g, function(x, item) { | ||||
| 		return {'&': '&', '"': '"', '<': '<', '>': '>', ''': "'"}[item]; | ||||
| 	}); | ||||
| } | ||||
|  | ||||
| // xmpp.js | ||||
| function XmlStreamParser() { | ||||
| 	this.buffer = ""; | ||||
| 	this._parsed = []; | ||||
| 	this.reset(); | ||||
| 	return this; | ||||
| } | ||||
|  | ||||
| XmlStreamParser.kText = "text"; | ||||
| XmlStreamParser.kElement = "element"; | ||||
| XmlStreamParser.kEndElement = "endElement"; | ||||
| XmlStreamParser.kAttributeName = "attributeName"; | ||||
| XmlStreamParser.kAttributeValue = "attributeValue"; | ||||
|  | ||||
| XmlStreamParser.prototype.reset = function() { | ||||
| 	this._state = XmlStreamParser.kText; | ||||
| 	this._attributes = {}; | ||||
| 	this._attributeName = ""; | ||||
| 	this._attributeValue = ""; | ||||
| 	this._attributeEquals = false; | ||||
| 	this._attributeQuote = ""; | ||||
| 	this._slash = false; | ||||
| 	this._value = ""; | ||||
| 	this._decl = false; | ||||
| } | ||||
|  | ||||
| XmlStreamParser.prototype.parse = function(data) { | ||||
| 	this._parsed = []; | ||||
|  | ||||
| 	for (var i = 0; i < data.length; i++) { | ||||
| 		var c = data.charAt(i); | ||||
| 		this.parseCharacter(c); | ||||
| 	} | ||||
|  | ||||
| 	return this._parsed; | ||||
| } | ||||
|  | ||||
| XmlStreamParser.prototype.flush = function() { | ||||
| 	var node = {type: this._state}; | ||||
| 	if (this._value) { | ||||
| 		node.value = xmlDecode(this._value); | ||||
| 	} | ||||
| 	if (this._attributes.length || this._state == XmlStreamParser.kElement) { | ||||
| 		node.attributes = this._attributes; | ||||
| 	} | ||||
| 	if (this._state != XmlStreamParser.kText || this._value) { | ||||
| 		this.emit(node); | ||||
| 	} | ||||
| 	this.reset(); | ||||
| } | ||||
|  | ||||
| XmlStreamParser.prototype.parseCharacter = function(c) { | ||||
| 	switch (this._state) { | ||||
| 	case XmlStreamParser.kText: | ||||
| 		if (c == '<') { | ||||
| 			this.flush(); | ||||
| 			this._state = XmlStreamParser.kElement; | ||||
| 		} else { | ||||
| 			this._value += c; | ||||
| 		} | ||||
| 		break; | ||||
| 	case XmlStreamParser.kElement: | ||||
| 	case XmlStreamParser.kEndElement: | ||||
| 		switch (c) { | ||||
| 		case '>': | ||||
| 			this.finishElement(); | ||||
| 			break; | ||||
| 		case '/': | ||||
| 			if (!this._value) { | ||||
| 				this._state = XmlStreamParser.kEndElement; | ||||
| 			} else if (!this._slash) { | ||||
| 				this._slash = true; | ||||
| 			} else { | ||||
| 				this._value += c; | ||||
| 			} | ||||
| 			break; | ||||
| 		case '?': | ||||
| 			if (!this._value) { | ||||
| 				this._decl = true; | ||||
| 			} else { | ||||
| 				this._value += '?'; | ||||
| 			} | ||||
| 			break; | ||||
| 		case ' ': | ||||
| 		case '\t': | ||||
| 		case '\r': | ||||
| 		case '\n': | ||||
| 			this._state = XmlStreamParser.kAttributeName; | ||||
| 			break; | ||||
| 		default: | ||||
| 			if (this._slash) { | ||||
| 				this._slash = false; | ||||
| 				this._value += '/'; | ||||
| 			} | ||||
| 			this._value += c; | ||||
| 			break; | ||||
| 		} | ||||
| 		break; | ||||
| 	case XmlStreamParser.kAttributeName: | ||||
| 		switch (c) { | ||||
| 		case ' ': | ||||
| 		case '\t': | ||||
| 		case '\r': | ||||
| 		case '\n': | ||||
| 			if (this._attributeName) { | ||||
| 				this._state = XmlStreamParser.kAttributeValue; | ||||
| 			} | ||||
| 			break; | ||||
| 		case '/': | ||||
| 			if (!this._slash) { | ||||
| 				this._slash = true; | ||||
| 			} else { | ||||
| 				this._value += '/'; | ||||
| 			} | ||||
| 			break; | ||||
| 		case '=': | ||||
| 			this._state = XmlStreamParser.kAttributeValue; | ||||
| 			break; | ||||
| 		case '>': | ||||
| 			if (this._attributeName) { | ||||
| 				this._attributes[this._attributeName] = null; | ||||
| 			} | ||||
| 			this._state = XmlStreamParser.kElement; | ||||
| 			this.finishElement(); | ||||
| 			break; | ||||
| 		default: | ||||
| 			this._attributeName += c; | ||||
| 			break; | ||||
| 		} | ||||
| 		break; | ||||
| 	case XmlStreamParser.kAttributeValue: | ||||
| 		switch (c) { | ||||
| 		case ' ': | ||||
| 		case '\t': | ||||
| 		case '\r': | ||||
| 		case '\n': | ||||
| 			if (this._attributeValue) { | ||||
| 				this._state = XmlStreamParser.kAttributeName; | ||||
| 			} | ||||
| 			break; | ||||
| 		case '"': | ||||
| 		case "'": | ||||
| 			if (!this._attributeValue && !this._attributeQuote) { | ||||
| 				this._attributeQuote = c; | ||||
| 			} else if (this._attributeQuote == c) { | ||||
| 				this._attributes[this._attributeName] = this._attributeValue; | ||||
| 				this._attributeName = ""; | ||||
| 				this._attributeValue = ""; | ||||
| 				this._attributeQuote = ""; | ||||
| 				this._state = XmlStreamParser.kAttributeName; | ||||
| 			} else { | ||||
| 				this._attributeValue += c; | ||||
| 			} | ||||
| 			break; | ||||
| 		case '>': | ||||
| 			this.finishElement(); | ||||
| 			break; | ||||
| 		default: | ||||
| 			this._attributeValue += c; | ||||
| 			break; | ||||
| 		} | ||||
| 		break; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| XmlStreamParser.prototype.finishElement = function() { | ||||
| 	if (this._decl) { | ||||
| 		this.reset(); | ||||
| 	} else { | ||||
| 		var value = this._value; | ||||
| 		var slash = this._slash; | ||||
| 		this.flush(); | ||||
| 		if (slash) { | ||||
| 			this._state = XmlStreamParser.kEndElement; | ||||
| 			this._value = value; | ||||
| 			this.flush(); | ||||
| 		} | ||||
| 	} | ||||
| 	this._state = XmlStreamParser.kText; | ||||
| } | ||||
|  | ||||
| XmlStreamParser.prototype.emit = function(node) { | ||||
| 	this._parsed.push(node); | ||||
| } | ||||
|  | ||||
| function XmlStanzaParser(depth) { | ||||
| 	this._depth = depth || 0; | ||||
| 	this._parsed = []; | ||||
| 	this._stack = []; | ||||
| 	this._stream = new XmlStreamParser(); | ||||
| 	return this; | ||||
| } | ||||
|  | ||||
| XmlStanzaParser.prototype.reset = function() { | ||||
| 	this._parsed = []; | ||||
| 	this._stack = []; | ||||
| 	this._stream.reset(); | ||||
| } | ||||
|  | ||||
| XmlStanzaParser.prototype.emit = function(stanza) { | ||||
| 	this._parsed.push(stanza); | ||||
| } | ||||
|  | ||||
| XmlStanzaParser.prototype.parse = function(data) { | ||||
| 	this._parsed = []; | ||||
| 	var nodes = this._stream.parse(data); | ||||
| 	for (var i = 0; i < nodes.length; i++) { | ||||
| 		this.parseNode(nodes[i]); | ||||
| 	} | ||||
| 	return this._parsed; | ||||
| } | ||||
|  | ||||
| XmlStanzaParser.prototype.parseNode = function(node) { | ||||
| 	switch (node.type) { | ||||
| 	case XmlStreamParser.kElement: | ||||
| 		this._stack.push({name: node.value, attributes: node.attributes, children: [], text: ""}); | ||||
| 		break; | ||||
| 	case XmlStreamParser.kEndElement: | ||||
| 		if (this._stack.length == 1 + this._depth) { | ||||
| 			this.emit(this._stack.pop()); | ||||
| 		} else { | ||||
| 			var last = this._stack.pop(); | ||||
| 			this._stack[this._stack.length - 1].children.push(last); | ||||
| 		} | ||||
| 		break; | ||||
| 	case XmlStreamParser.kText: | ||||
| 		if (this._stack) { | ||||
| 			this._stack[this._stack.length - 1].text += node.value; | ||||
| 		} | ||||
| 		break; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // end xmpp.js | ||||
|  | ||||
| var gPingCount = 0; | ||||
|  | ||||
| class XmppService { | ||||
| 	constructor(options) { | ||||
| 		let self = this; | ||||
| 		self._callback = options.callback; | ||||
| 		self._conversations = {}; | ||||
|  | ||||
| 		network.newConnection().then(function(socket) { | ||||
| 			self._socket = socket; | ||||
| 			return self._connect(options); | ||||
| 		}).catch(self._reportError); | ||||
| 	} | ||||
|  | ||||
| 	sendMessage(to, message) { | ||||
| 		this._socket.write("<message type='groupchat' to='" + xmlEncode(to) + "'><body>" + xmlEncode(message) + "</body></message>"); | ||||
| 	} | ||||
|  | ||||
| 	getConversations() { | ||||
| 		return Object.keys(this._conversations); | ||||
| 	} | ||||
|  | ||||
| 	getParticipants(conversation) { | ||||
| 		let result; | ||||
| 		if (this._conversations[conversation]) { | ||||
| 			result = this._conversations[conversation].participants; | ||||
| 		} | ||||
| 		return result; | ||||
| 	} | ||||
|  | ||||
| 	getHistory(conversation) { | ||||
| 		let result; | ||||
| 		if (this._conversations[conversation]) { | ||||
| 			result = this._conversations[conversation].history; | ||||
| 		} | ||||
| 		return result; | ||||
| 	} | ||||
|  | ||||
| 	_connect(options) { | ||||
| 		let self = this; | ||||
| 		var kTrustedCertificate = "-----BEGIN CERTIFICATE-----\n" + | ||||
| 			"MIICqjCCAhOgAwIBAgIJAPEhMguftPdoMA0GCSqGSIb3DQEBCwUAMG4xCzAJBgNV\n" + | ||||
| 			"BAYTAlVTMQ4wDAYDVQQIDAVUZXhhczEPMA0GA1UEBwwGQXVzdGluMRswGQYDVQQK\n" + | ||||
| 			"DBJUcm91YmxlIEltcGFjdCBMTEMxITAfBgNVBAMMGGphYmJlci50cm91YmxlaW1w\n" + | ||||
| 			"YWN0LmNvbTAeFw0xNDEyMjYwMzU5NDRaFw0yNDEyMjMwMzU5NDRaMG4xCzAJBgNV\n" + | ||||
| 			"BAYTAlVTMQ4wDAYDVQQIDAVUZXhhczEPMA0GA1UEBwwGQXVzdGluMRswGQYDVQQK\n" + | ||||
| 			"DBJUcm91YmxlIEltcGFjdCBMTEMxITAfBgNVBAMMGGphYmJlci50cm91YmxlaW1w\n" + | ||||
| 			"YWN0LmNvbTCBnzANBgkqhkiG9w0BAQEFAAOBjQAwgYkCgYEAueniASgCpF7mQFGt\n" + | ||||
| 			"TycOhMt9VMetFwwDkwVglvO+VKq8JWxWkJaCWm8YYacG6+zn4RlV3zVQhrAcReTU\n" + | ||||
| 			"pPQAe+28wJdqVt/HPyfcwJtLKUEL7Nk5N8mY2s6yyBVvMn9e7Yt/fnv7pOCpcmBi\n" + | ||||
| 			"kuLlwSGEfMnDskt8kH4coidP4w0CAwEAAaNQME4wHQYDVR0OBBYEFOztZhuuqXrN\n" + | ||||
| 			"yUnPo/9aoNNb/o2CMB8GA1UdIwQYMBaAFOztZhuuqXrNyUnPo/9aoNNb/o2CMAwG\n" + | ||||
| 			"A1UdEwQFMAMBAf8wDQYJKoZIhvcNAQELBQADgYEAgK/7yoGEHeG95i6E1A8ZBkeL\n" + | ||||
| 			"monKMys3RxnJciuFdBrUcvymsgOTrAGvatPXatNbHQ/eY8LnkKHtf0pCCs0B/xST\n" + | ||||
| 			"DTO3KdlNCXApMUieFPjVggRzikbmbPCvtTt2BzqQKzVqubf9eM+kbsD7Pkgycm5+\n" + | ||||
| 			"q46TZws0oz5lAvklIgo=\n" + | ||||
| 			"-----END CERTIFICATE-----"; | ||||
| 		var resource = options.resource || "tildefriends"; | ||||
| 		let userName = options.userName; | ||||
| 		let password = options.password; | ||||
| 		let server = options.server; | ||||
| 		self._socket.connect("jabber.troubleimpact.com", 5222).then(function() { | ||||
| 			print("actually connected"); | ||||
| 			self._callback({action: "connected"}); | ||||
| 			print("wtf"); | ||||
| 			var parse = new XmlStanzaParser(1); | ||||
| 			self._socket.write("<?xml version='1.0'?>"); | ||||
| 			self._socket.write("<stream:stream to='" + xmlEncode(server) + "' xmlns='jabber:client' xmlns:stream='http://etherx.jabber.org/streams' version='1.0'>"); | ||||
|  | ||||
| 			var started = false; | ||||
| 			var authenticated = false; | ||||
| 			self._socket.onError(self._reportError); | ||||
| 			self._socket.read(function(data) { | ||||
| 				try { | ||||
| 					if (!data) { | ||||
| 						self._callback({action: "disconnected"}); | ||||
| 						return; | ||||
| 					} | ||||
| 					parse.parse(data).forEach(function(stanza) { | ||||
| 						if (stanza.name == "stream:features") { | ||||
| 							if (!started) { | ||||
| 								self._socket.write("<starttls xmlns='urn:ietf:params:xml:ns:xmpp-tls'/>"); | ||||
| 							} else if (!authenticated) { | ||||
| 								self._socket.write("<auth xmlns='urn:ietf:params:xml:ns:xmpp-sasl' mechanism='DIGEST-MD5'/>"); | ||||
| 							} else { | ||||
| 								self._socket.write("<iq type='set' id='bind0'><bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'><resource>" + resource + "</resource></bind></iq>"); | ||||
| 							} | ||||
| 						} else if (stanza.name == "proceed") { | ||||
| 							if (!started) { | ||||
| 								started = true; | ||||
| 								self._socket.addTrustedCertificate(kTrustedCertificate); | ||||
| 								self._socket.startTls().then(function() { | ||||
| 									parse.reset(); | ||||
| 									self._socket.write("<stream:stream to='" + xmlEncode(server) + "' xmlns='jabber:client' xmlns:stream='http://etherx.jabber.org/streams' version='1.0'>"); | ||||
| 								}).catch(self._reportError); | ||||
| 							} | ||||
| 						} else if (stanza.name == "success") { | ||||
| 							authenticated = true; | ||||
| 							self._socket.write("<?xml version='1.0'?>"); | ||||
| 							self._socket.write("<stream:stream to='" + xmlEncode(server) + "' xmlns='jabber:client' xmlns:stream='http://etherx.jabber.org/streams' version='1.0'>"); | ||||
| 							parse.reset(); | ||||
| 						} else if (stanza.name == "iq") { | ||||
| 							if (stanza.attributes.id == "bind0") { | ||||
| 								self._socket.write("<iq type='set' id='session0'><session xmlns='urn:ietf:params:xml:ns:xmpp-session'/></iq>"); | ||||
| 							} else if (stanza.attributes.id == "session0") { | ||||
| 								self._socket.write("<presence to='chadhappyfuntime@conference.jabber.troubleimpact.com/" + userName + "'><priority>1</priority><x xmlns='http://jabber.org/protocol/muc'/></presence>"); | ||||
| 								self._schedulePing(); | ||||
| 								self._conversations["chadhappyfuntime@conference.jabber.troubleimpact.com"] = {participants: [], history: []}; | ||||
| 							} else if (stanza.attributes.id == "ping" + gPingCount) { | ||||
| 								// Ping response. | ||||
| 							} else { | ||||
| 								self._callback({ | ||||
| 									action: "unknown", | ||||
| 									stanza: stanza, | ||||
| 								}); | ||||
| 							} | ||||
| 						} else if (stanza.name == "message") { | ||||
| 							let message = self._convertMessage(stanza); | ||||
| 							self._conversations[message.conversation].history.push(message); | ||||
| 							self._callback(message); | ||||
| 						} else if (stanza.name == "challenge") { | ||||
| 							var challenge = Base64.decode(stanza.text); | ||||
| 							var parts = challenge.split(','); | ||||
| 							challenge = {}; | ||||
| 							for (var i = 0; i < parts.length; i++) { | ||||
| 								var equals = parts[i].indexOf("="); | ||||
| 								if (equals != -1) { | ||||
| 									var key = parts[i].substring(0, equals); | ||||
| 									var value = parts[i].substring(equals + 1); | ||||
| 									if (value.length > 2 && value.charAt(0) == '"' && value.charAt(value.length - 1) == '"') { | ||||
| 										value = value.substring(1, value.length - 1); | ||||
| 									} | ||||
| 									challenge[key] = value; | ||||
| 								} | ||||
| 							} | ||||
| 							if (challenge.rspauth) { | ||||
| 								self._socket.write("<response xmlns='urn:ietf:params:xml:ns:xmpp-sasl'/>"); | ||||
| 							} else { | ||||
| 								var realm = server; | ||||
| 								var cnonce = Base64.encode(new Date().toString()); | ||||
| 								var x = userName + ":" + realm + ":" + password; | ||||
| 								var y = raw_md5(x); | ||||
| 								var a1 = y + ":" + challenge.nonce + ":" + cnonce; | ||||
| 								var digestUri = "xmpp/" + realm; | ||||
| 								var a2 = "AUTHENTICATE:" + digestUri; | ||||
| 								var ha1 = md5(a1); | ||||
| 								var ha2 = md5(a2); | ||||
| 								var nc = "00000001"; | ||||
| 								var kd = ha1 + ":" + challenge.nonce + ":" + nc + ":" + cnonce + ":" + challenge.qop + ":" + ha2; | ||||
| 								var response = md5(kd); | ||||
| 								var value = Base64.encode('username="' + userName + '",realm="' + realm + '",nonce="' + challenge.nonce + '",cnonce="' + cnonce + '",nc=' + nc + ',qop=' + challenge.qop + ',digest-uri="' + digestUri + '",response=' + response + ',charset=utf-8'); | ||||
| 								self._socket.write("<response xmlns='urn:ietf:params:xml:ns:xmpp-sasl'>" + value + "</response>"); | ||||
| 							} | ||||
| 						} else if (stanza.name == "presence") { | ||||
| 							let name = stanza.attributes.from.split('/', 2)[1]; | ||||
| 							let conversation = stanza.attributes.from.split('/', 2)[0]; | ||||
| 							let leaving = stanza.attributes.type == "unavailable"; | ||||
| 							if (leaving) { | ||||
| 								self._conversations[conversation].participants.remove(name); | ||||
| 							} else { | ||||
| 								if (self._conversations[conversation].participants.indexOf(name) == -1) { | ||||
| 									self._conversations[conversation].participants.push(name); | ||||
| 								} | ||||
| 							} | ||||
| 							self._callback({ | ||||
| 								action: "presence", | ||||
| 								name: name, | ||||
| 								jid: stanza.attributes.from, | ||||
| 								type: stanza.attributes.type, | ||||
| 							}); | ||||
| 						} else { | ||||
| 							self._callback({ | ||||
| 								action: "unknown", | ||||
| 								stanza: stanza, | ||||
| 							}); | ||||
| 						} | ||||
| 					}); | ||||
| 				} catch (error) { | ||||
| 					self._reportError(error); | ||||
| 				} | ||||
| 			}); | ||||
| 		}).catch(self._reportError); | ||||
| 	} | ||||
|  | ||||
| 	disconnect() { | ||||
| 		self._socket.write("</stream>"); | ||||
| 		self._socket.close(); | ||||
| 		delete gSessions[self._name]; | ||||
| 	} | ||||
|  | ||||
| 	_reportError(error) { | ||||
| 		this._callback({ | ||||
| 			action: "error", | ||||
| 			error: error, | ||||
| 		}).catch(function(error) { | ||||
| 			print(error); | ||||
| 		}); | ||||
| 	} | ||||
|  | ||||
| 	_convertMessage(stanza) { | ||||
| 		let self = this; | ||||
| 		let text; | ||||
| 		let now = new Date().toString(); | ||||
| 		for (var i in stanza.children) { | ||||
| 			if (stanza.children[i].name == "body") { | ||||
| 				text = stanza.children[i].text; | ||||
| 			} | ||||
| 			if (stanza.children[i].name == "delay") { | ||||
| 				now = new Date(stanza.children[i].attributes.stamp).toString(); | ||||
| 			} | ||||
| 		} | ||||
| 		let from = stanza.attributes.from || "unknown"; | ||||
| 		if (from && from.indexOf('/') != -1) { | ||||
| 			from = from.split("/")[1]; | ||||
| 		} | ||||
| 		let conversation = from; | ||||
| 		if (stanza.attributes.type == "groupchat") { | ||||
| 			if (self._conversations[stanza.attributes.to.split("/")[0]]) { | ||||
| 				conversation = stanza.attributes.to.split("/")[0]; | ||||
| 			} else if (self._conversations[stanza.attributes.from.split("/")[0]]) { | ||||
| 				conversation = stanza.attributes.from.split("/")[0]; | ||||
| 			} | ||||
| 		} | ||||
| 		let message = { | ||||
| 			action: "message", | ||||
| 			from: from, | ||||
| 			conversation: conversation, | ||||
| 			message: text, | ||||
| 			stanza: stanza, | ||||
| 			timestamp: now, | ||||
| 		}; | ||||
| 		return message; | ||||
| 	} | ||||
|  | ||||
| 	_schedulePing() { | ||||
| 		let self = this; | ||||
| 		setTimeout(function() { | ||||
| 			self._socket.write("<iq type='get' id='ping" + (++gPingCount) + "'><ping xmlns='urn:xmpp:ping'/></iq>"); | ||||
| 			self._schedulePing(); | ||||
| 		}, 60000); | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| let gSessions = {}; | ||||
|  | ||||
| core.register("onMessage", function(sender, options) { | ||||
| 	let service = gSessions[options.name]; | ||||
| 	if (!service) { | ||||
| 		service = new XmppService(options); | ||||
| 		gSessions[options.name] = service; | ||||
| 	} else { | ||||
| 		service._callback = options.callback; | ||||
| 	} | ||||
| 	return { | ||||
| 		sendMessage: service.sendMessage.bind(service), | ||||
| 		getConversations: service.getConversations.bind(service), | ||||
| 		getHistory: service.getHistory.bind(service), | ||||
| 		getParticipants: service.getParticipants.bind(service), | ||||
| 		disconnect: service.disconnect.bind(service), | ||||
| 	}; | ||||
| }); | ||||
		Reference in New Issue
	
	Block a user