// // Copyright 2012 "Leberwurscht" <leberwurscht@hoegners.de> // // This file is dual-licensed under the MIT license (see MIT.txt) and the AGPL license (see jappix/COPYING). // function jappixmini_addon_xor(str1, str2) { if (str1.length != str2.length) throw "not same length"; var encoded = ""; for (var i=0; i<str1.length;i++) { var a = str1.charCodeAt(i); var b = str2.charCodeAt(i); var c = a ^ b; encoded += String.fromCharCode(c); } return encoded; } function jappixmini_addon_set_client_secret(password) { if (!password) return; var salt1 = "h8doCRekWto0njyQohKpdx6BN0UTyC6N"; var salt2 = "jdX8OwFC1kWAq3s9uOyAcE8g3UNNO5t3"; var client_secret1 = str_sha1(salt1+password); var client_secret2 = str_sha1(salt2+password); var client_secret = client_secret1 + client_secret2; setPersistent('jappix-mini', 'client-secret', client_secret); console.log("client secret set"); } function jappixmini_addon_get_client_secret(callback) { var client_secret = getPersistent('jappix-mini', 'client-secret'); if (client_secret===null) { var div = document.getElementById("#jappixmini-password-query-div"); if (!div) { div = $('<div id="jappixmini-password-query-div" style="position:fixed;padding:1em;background-color:#F00;color:#fff;top:50px;left:50px;">Retype your Friendica password for chatting:<br></div>'); var input = $('<input type="password" id="jappixmini-password-query-input">') div.append(input); var button = $('<input type="button" value="OK" id="jappixmini-password-query-button">'); div.append(button); $("body").append(div); } button.click(function(){ var password = $("#jappixmini-password-query-input").val(); jappixmini_addon_set_client_secret(password); div.remove(); var client_secret = getPersistent('jappix-mini', 'client-secret'); callback(client_secret); }); } else { callback(client_secret); } } function jappixmini_addon_encrypt_password(password, callback) { jappixmini_addon_get_client_secret(function(client_secret){ // add \0 to password until it has the same length as secret if (password.length>client_secret.length-1) throw "password too long"; while (password.length<client_secret.length) { password += "\0"; } // xor password with secret var encrypted_password = jappixmini_addon_xor(client_secret, password); encrypted_password = encodeURI(encrypted_password) callback(encrypted_password); }); } function jappixmini_addon_decrypt_password(encrypted_password, callback) { encrypted_password = decodeURI(encrypted_password); jappixmini_addon_get_client_secret(function(client_secret){ // xor password with secret var password = jappixmini_addon_xor(client_secret, encrypted_password); // remove \0 var first_null = password.indexOf("\0") if (first_null==-1) throw "Decrypted password does not contain \\0"; password = password.substr(0, first_null); callback(password); }); } function jappixmini_manage_roster(contacts, contacts_hash, autoapprove, autosubscribe) { // listen for subscriptions con.registerHandler('presence',function(presence){ var type = presence.getType(); if (type != "subscribe") return; var from = fullXID(getStanzaFrom(presence)); var xid = bareXID(from); var pstatus = presence.getStatus(); var approve; if (autoapprove && contacts[xid]!==undefined) { // approve known address approve = true; console.log("Approve known Friendica contact "+xid+"."); } else if (autoapprove && pstatus && pstatus.indexOf("Friendica")!=-1) { // Unknown address claims to be a Friendica contact. // This is probably because the other side knows our // address, but we do not know the other side yet. // But it's only a matter of time, so wait - do not // approve yet and do not annoy the user by asking. approve = false; console.log("Do not approve unknown Friendica contact "+xid+" - wait instead."); } else { // In all other cases, ask the user. var message = "Accept "+xid+" for chat?"; if (pstatus) message += "\n\nStatus:\n"+pstatus; approve = confirm(message); // do not ask any more if (!approve) sendSubscribe(xid, "unsubscribed"); } if (approve) { var name = contacts[xid]; if (!name) name = xid; acceptSubscribe(xid, name); console.log("Accepted "+xid+" ("+name+") for chat."); } }); // autosubscribe if (!autosubscribe) return; var stored_hash = getPersistent("jappix-mini", "contacts-hash"); var contacts_changed = (stored_hash != contacts_hash); // stored_hash gets updated later if everything was successful if (!contacts_changed) return; console.log("Start autosubscribe."); var get_roster = new JSJaCIQ(); get_roster.setType('get'); get_roster.setQuery(NS_ROSTER); con.send(get_roster, function(iq){ var handleXML = iq.getQuery(); // filter out contacts that are already in the roster $(handleXML).find('item').each(function() { var node = $(this); var xid = node.attr("jid"); var name = node.attr("name"); var subscription = node.attr("subscription"); // ignore accounts that are not in the list if (contacts[xid]===undefined) return; // add to Friendica group or change name if necessary var groups = []; var group_missing = false; node.find('group').each(function() { var group_text = $(this).text(); if (group_text) groups.push(group_text); }); if ($.inArray("Friendica", groups)==-1) { group_missing = true; groups.push("Friendica"); } if (group_missing || name!=contacts[xid]) { sendRoster(xid, null, contacts[xid], groups); console.log("Added "+xid+" to Friendica group and set name to "+contacts[xid]+"."); } // authorize if necessary if (subscription=="to") { sendSubscribe(xid, 'subscribed'); console.log("Authorized "+xid+" automatically."); } // remove from list delete contacts[xid]; }); // go through remaining contacts for (var xid in contacts) {if(!contacts.hasOwnProperty(xid)) continue; // subscribe var presence = new JSJaCPresence(); presence.setTo(xid); presence.setType("subscribe"); // must contain the word "~Friendica" so the other side knows // how to handle this presence.setStatus("I'm "+MINI_NICKNAME+" from ~Friendica.\n[machine-generated message]"); con.send(presence); console.log("Subscribed to "+xid+" automatically."); // add to roster var iq = new JSJaCIQ(); iq.setType('set'); var iqQuery = iq.setQuery(NS_ROSTER); var item = iqQuery.appendChild(iq.buildNode('item', {'xmlns': NS_ROSTER, 'jid': xid})); item.setAttribute('name', contacts[xid]); item.appendChild(iq.buildNode('group', {'xmlns': NS_ROSTER}, "Friendica")); con.send(iq); console.log("Added "+xid+" ("+contacts[xid]+") to roster."); } setPersistent("jappix-mini", "contacts-hash", contacts_hash); console.log("Autosubscribe done."); }); } function jappixmini_addon_subscribe() { if (!window.con) { alert("Not connected."); return; } var xid = prompt("Jabber address"); sendSubscribe(xid, "subscribe"); } function jappixmini_addon_start(server, username, proxy, bosh, encrypted, password, nickname, contacts, contacts_hash, autoapprove, autosubscribe) { var handler = function(password){ // check if settings have changed, reinitialize jappix mini if this is the case var settings_identifier = str_sha1(server); settings_identifier += str_sha1(username); settings_identifier += str_sha1(proxy); settings_identifier += str_sha1(bosh); settings_identifier += str_sha1(password); settings_identifier += str_sha1(nickname); var saved_identifier = getDB("jappix-mini", "settings-identifier"); if (saved_identifier != settings_identifier) { disconnectMini(); removeDB('jappix-mini', 'dom'); removePersistent("jappix-mini", "contacts-hash"); } setDB("jappix-mini", "settings-identifier", settings_identifier); // set HOST_BOSH if (proxy) HOST_BOSH = proxy+"?host_bosh="+encodeURI(bosh); else HOST_BOSH = bosh; // start jappix mini MINI_NICKNAME = nickname; LOCK_HOST = "off"; launchMini(true, false, server, username, password); // increase priority over other Jabber clients - does not seem to work? var priority = 101; presenceMini(null,null,priority); jappixmini_manage_roster(contacts, contacts_hash, autoapprove, autosubscribe) } // decrypt password if necessary if (encrypted) jappixmini_addon_decrypt_password(password, handler); else handler(password); }