define([ 'json.sortify', '/common/userObject.js', '/common/migrate-user-object.js', '/common/common-hash.js', '/common/common-util.js', '/common/common-constants.js', '/common/common-feedback.js', '/common/common-realtime.js', '/common/common-messaging.js', '/common/common-messenger.js', '/common/outer/chainpad-netflux-worker.js', '/common/outer/network-config.js', '/customize/application_config.js', '/bower_components/chainpad-crypto/crypto.js?v=0.1.5', '/bower_components/chainpad/chainpad.dist.js', '/bower_components/chainpad-listmap/chainpad-listmap.js', '/bower_components/nthen/index.js', '/bower_components/saferphore/index.js', ], function (Sortify, UserObject, Migrate, Hash, Util, Constants, Feedback, Realtime, Messaging, Messenger, CpNfWorker, NetConfig, AppConfig, Crypto, ChainPad, Listmap, nThen, Saferphore) { var Store = {}; var postMessage = function () {}; var storeHash; var store = window.CryptPad_AsyncStore = {}; var onSync = function (cb) { Realtime.whenRealtimeSyncs(store.realtime, cb); }; Store.get = function (key, cb) { cb(Util.find(store.proxy, key)); }; Store.set = function (data, cb) { var path = data.key.slice(); var key = path.pop(); var obj = Util.find(store.proxy, path); if (!obj || typeof(obj) !== "object") { return void cb({error: 'INVALID_PATH'}); } if (typeof data.value === "undefined") { delete obj[key]; } else { obj[key] = data.value; } onSync(cb); }; Store.hasSigningKeys = function () { if (!store.proxy) { return; } return typeof(store.proxy.edPrivate) === 'string' && typeof(store.proxy.edPublic) === 'string'; }; Store.hasCurveKeys = function () { if (!store.proxy) { return; } return typeof(store.proxy.curvePrivate) === 'string' && typeof(store.proxy.curvePublic) === 'string'; }; var getUserChannelList = function () { // start with your userHash... var userHash = storeHash; if (!userHash) { return null; } var userParsedHash = Hash.parseTypeHash('drive', userHash); var userChannel = userParsedHash && userParsedHash.channel; if (!userChannel) { return null; } // Get the list of pads' channel ID in your drive // This list is filtered so that it doesn't include pad owned by other users (you should // not pin these pads) var files = store.userObject.getFiles([store.userObject.FILES_DATA]); var edPublic = store.proxy.edPublic; var list = files.map(function (id) { var d = store.userObject.getFileData(id); if (d.owners && d.owners.length && edPublic && d.owners.indexOf(edPublic) === -1) { return; } return Hash.hrefToHexChannelId(d.href); }) .filter(function (x) { return x; }); // Get the avatar var profile = store.proxy.profile; if (profile) { var profileChan = profile.edit ? Hash.hrefToHexChannelId('/profile/#' + profile.edit) : null; if (profileChan) { list.push(profileChan); } var avatarChan = profile.avatar ? Hash.hrefToHexChannelId(profile.avatar) : null; if (avatarChan) { list.push(avatarChan); } } if (store.proxy.friends) { var fList = Messaging.getFriendChannelsList(store.proxy); list = list.concat(fList); } list.push(Util.base64ToHex(userChannel)); list.sort(); return list; }; var getExpirableChannelList = function () { var list = []; store.userObject.getFiles([store.userObject.FILES_DATA]).forEach(function (id) { var data = store.userObject.getFileData(id); var edPublic = store.proxy.edPublic; // Push channels owned by someone else or channel that should have expired // because of the expiration time if ((data.owners && data.owners.length && data.owners.indexOf(edPublic) === -1) || (data.expire && data.expire < (+new Date()))) { list.push(Hash.hrefToHexChannelId(data.href)); } }); return list; }; var getCanonicalChannelList = function (expirable) { var list = expirable ? getExpirableChannelList() : getUserChannelList(); return Util.deduplicateString(list).sort(); }; ////////////////////////////////////////////////////////////////// /////////////////////// RPC ////////////////////////////////////// ////////////////////////////////////////////////////////////////// Store.pinPads = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } if (typeof(cb) !== 'function') { console.error('expected a callback'); } store.rpc.pin(data, function (e, hash) { if (e) { return void cb({error: e}); } cb({hash: hash}); }); }; Store.unpinPads = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } store.rpc.unpin(data, function (e, hash) { if (e) { return void cb({error: e}); } cb({hash: hash}); }); }; var account = {}; Store.getPinnedUsage = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } store.rpc.getFileListSize(function (err, bytes) { if (typeof(bytes) === 'number') { account.usage = bytes; } cb({bytes: bytes}); }); }; // Update for all users from accounts and return current user limits Store.updatePinLimit = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } store.rpc.updatePinLimits(function (e, limit, plan, note) { if (e) { return void cb({error: e}); } account.limit = limit; account.plan = plan; account.note = note; cb(account); }); }; // Get current user limits Store.getPinLimit = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } var ALWAYS_REVALIDATE = true; if (ALWAYS_REVALIDATE || typeof(account.limit) !== 'number' || typeof(account.plan) !== 'string' || typeof(account.note) !== 'string') { return void store.rpc.getLimit(function (e, limit, plan, note) { if (e) { return void cb({error: e}); } account.limit = limit; account.plan = plan; account.note = note; cb(account); }); } cb(account); }; Store.clearOwnedChannel = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } store.rpc.clearOwnedChannel(data, function (err) { cb({error:err}); }); }; Store.removeOwnedChannel = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } store.rpc.removeOwnedChannel(data, function (err) { cb({error:err}); }); }; var arePinsSynced = function (cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } var list = getCanonicalChannelList(false); var local = Hash.hashChannelList(list); store.rpc.getServerHash(function (e, hash) { if (e) { return void cb(e); } cb(null, hash === local); }); }; var resetPins = function (cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } var list = getCanonicalChannelList(false); store.rpc.reset(list, function (e, hash) { if (e) { return void cb(e); } cb(null, hash); }); }; Store.uploadComplete = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } store.rpc.uploadComplete(function (err, res) { if (err) { return void cb({error:err}); } cb(res); }); }; Store.uploadStatus = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } store.rpc.uploadStatus(data.size, function (err, res) { if (err) { return void cb({error:err}); } cb(res); }); }; Store.uploadCancel = function (data, cb) { if (!store.rpc) { return void cb({error: 'RPC_NOT_READY'}); } store.rpc.uploadCancel(function (err, res) { if (err) { return void cb({error:err}); } cb(res); }); }; Store.uploadChunk = function (data, cb) { store.rpc.send.unauthenticated('UPLOAD', data.chunk, function (e, msg) { cb({ error: e, msg: msg }); }); }; Store.initRpc = function (data, cb) { require(['/common/pinpad.js'], function (Pinpad) { Pinpad.create(store.network, store.proxy, function (e, call) { if (e) { return void cb({error: e}); } store.rpc = call; Store.getPinLimit(null, function (obj) { if (obj.error) { console.error(obj.error); } account.limit = obj.limit; account.plan = obj.plan; account.note = obj.note; cb(obj); }); arePinsSynced(function (err, yes) { if (!yes) { resetPins(function (err) { if (err) { return console.error(err); } console.log('RESET DONE'); }); } }); }); }); }; ////////////////////////////////////////////////////////////////// ////////////////// ANON RPC ////////////////////////////////////// ////////////////////////////////////////////////////////////////// Store.anonRpcMsg = function (data, cb) { if (!store.anon_rpc) { return void cb({error: 'ANON_RPC_NOT_READY'}); } store.anon_rpc.send(data.msg, data.data, function (err, res) { if (err) { return void cb({error: err}); } cb(res); }); }; Store.getFileSize = function (data, cb) { if (!store.anon_rpc) { return void cb({error: 'ANON_RPC_NOT_READY'}); } var channelId = Hash.hrefToHexChannelId(data.href); store.anon_rpc.send("GET_FILE_SIZE", channelId, function (e, response) { if (e) { return void cb({error: e}); } if (response && response.length && typeof(response[0]) === 'number') { return void cb({size: response[0]}); } else { cb({error: 'INVALID_RESPONSE'}); } }); }; Store.isNewChannel = function (data, cb) { if (!store.anon_rpc) { return void cb({error: 'ANON_RPC_NOT_READY'}); } var channelId = Hash.hrefToHexChannelId(data.href); store.anon_rpc.send("IS_NEW_CHANNEL", channelId, function (e, response) { if (e) { return void cb({error: e}); } if (response && response.length && typeof(response[0]) === 'boolean') { return void cb({ isNew: response[0] }); } else { cb({error: 'INVALID_RESPONSE'}); } }); }; Store.getMultipleFileSize = function (data, cb) { if (!store.anon_rpc) { return void cb({error: 'ANON_RPC_NOT_READY'}); } if (!Array.isArray(data.files)) { return void cb({error: 'INVALID_FILE_LIST'}); } store.anon_rpc.send('GET_MULTIPLE_FILE_SIZE', data.files, function (e, res) { if (e) { return void cb({error: e}); } if (res && res.length && typeof(res[0]) === 'object') { cb({size: res[0]}); } else { cb({error: 'UNEXPECTED_RESPONSE'}); } }); }; Store.getDeletedPads = function (data, cb) { if (!store.anon_rpc) { return void cb({error: 'ANON_RPC_NOT_READY'}); } var list = getCanonicalChannelList(true); if (!Array.isArray(list)) { return void cb({error: 'INVALID_FILE_LIST'}); } store.anon_rpc.send('GET_DELETED_PADS', list, function (e, res) { if (e) { return void cb({error: e}); } if (res && res.length && Array.isArray(res[0])) { cb(res[0]); } else { cb({error: 'UNEXPECTED_RESPONSE'}); } }); }; Store.initAnonRpc = function (data, cb) { require([ '/common/rpc.js', ], function (Rpc) { Rpc.createAnonymous(store.network, function (e, call) { if (e) { return void cb({error: e}); } store.anon_rpc = call; cb(); }); }); }; ////////////////////////////////////////////////////////////////// /////////////////////// Store //////////////////////////////////// ////////////////////////////////////////////////////////////////// // Get the metadata for sframe-common-outer Store.getMetadata = function (data, cb) { var disableThumbnails = Util.find(store.proxy, ['settings', 'general', 'disableThumbnails']); var metadata = { // "user" is shared with everybody via the userlist user: { name: store.proxy[Constants.displayNameKey] || "", uid: store.proxy.uid, avatar: Util.find(store.proxy, ['profile', 'avatar']), profile: Util.find(store.proxy, ['profile', 'view']), curvePublic: store.proxy.curvePublic, }, // "priv" is not shared with other users but is needed by the apps priv: { edPublic: store.proxy.edPublic, friends: store.proxy.friends || {}, settings: store.proxy.settings, thumbnails: disableThumbnails === false } }; cb(JSON.parse(JSON.stringify(metadata))); }; var makePad = function (href, title) { var now = +new Date(); return { href: href, atime: now, ctime: now, title: title || Hash.getDefaultName(Hash.parsePadUrl(href)), }; }; Store.addPad = function (data, cb) { if (!data.href) { return void cb({error:'NO_HREF'}); } var pad = makePad(data.href, data.title); if (data.owners) { pad.owners = data.owners; } if (data.expire) { pad.expire = data.expire; } store.userObject.pushData(pad, function (e, id) { if (e) { return void cb({error: "Error while adding a template:"+ e}); } var path = data.path || ['root']; store.userObject.add(id, path); onSync(cb); }); }; var getOwnedPads = function () { var list = []; store.userObject.getFiles([store.userObject.FILES_DATA]).forEach(function (id) { var data = store.userObject.getFileData(id); var edPublic = store.proxy.edPublic; // Push channels owned by someone else or channel that should have expired // because of the expiration time if (data.owners && data.owners.length === 1 && data.owners.indexOf(edPublic) !== -1) { list.push(Hash.hrefToHexChannelId(data.href)); } }); return list; }; var removeOwnedPads = function (waitFor) { // Delete owned pads var ownedPads = getOwnedPads(); var sem = Saferphore.create(10); ownedPads.forEach(function (c) { var w = waitFor(); sem.take(function (give) { Store.removeOwnedChannel(c, give(function (obj) { if (obj && obj.error) { console.error(obj.error); } w(); })); }); }); }; Store.deleteAccount = function (data, cb) { var edPublic = store.proxy.edPublic; var secret = Hash.getSecrets('drive', storeHash); Store.anonRpcMsg({ msg: 'GET_METADATA', data: secret.channel }, function (data) { var metadata = data[0]; // Owned drive if (metadata && metadata.owners && metadata.owners.length === 1 && metadata.owners.indexOf(edPublic) !== -1) { nThen(function (waitFor) { var token = Math.floor(Math.random()*Number.MAX_SAFE_INTEGER); store.proxy[Constants.tokenKey] = token; postMessage("DELETE_ACCOUNT", token, waitFor()); }).nThen(function (waitFor) { removeOwnedPads(waitFor); }).nThen(function (waitFor) { // Delete Pin Store store.rpc.removePins(waitFor(function (err) { if (err) { console.error(err); } })); }).nThen(function (waitFor) { // Delete Drive Store.removeOwnedChannel(secret.channel, waitFor()); }).nThen(function () { store.network.disconnect(); cb({ state: true }); }); return; } // Not owned drive var toSign = { intent: 'Please delete my account.' }; toSign.drive = secret.channel; toSign.edPublic = edPublic; var signKey = Crypto.Nacl.util.decodeBase64(store.proxy.edPrivate); var proof = Crypto.Nacl.sign.detached(Crypto.Nacl.util.decodeUTF8(Sortify(toSign)), signKey); var check = Crypto.Nacl.sign.detached.verify(Crypto.Nacl.util.decodeUTF8(Sortify(toSign)), proof, Crypto.Nacl.util.decodeBase64(edPublic)); if (!check) { console.error('signed message failed verification'); } var proofTxt = Crypto.Nacl.util.encodeBase64(proof); cb({ proof: proofTxt, toSign: JSON.parse(Sortify(toSign)) }); }); }; /** * add a "What is CryptPad?" pad in the drive * data * - driveReadme * - driveReadmeTitle */ Store.createReadme = function (data, cb) { require(['/common/cryptget.js'], function (Crypt) { var hash = Hash.createRandomHash(); Crypt.put(hash, data.driveReadme, function (e) { if (e) { return void cb({ error: "Error while creating the default pad:"+ e}); } var href = '/pad/#' + hash; var fileData = { href: href, title: data.driveReadmeTitle, atime: +new Date(), ctime: +new Date() }; store.userObject.pushData(fileData, function (e, id) { if (e) { return void cb({ error: "Error while creating the default pad:"+ e}); } store.userObject.add(id); onSync(cb); }); }); }); }; /** * Merge the anonymous drive into the user drive at registration * data * - anonHash */ Store.migrateAnonDrive = function (data, cb) { require(['/common/mergeDrive.js'], function (Merge) { var hash = data.anonHash; Merge.anonDriveIntoUser(store, hash, cb); }); }; var getAttributeObject = function (attr) { if (typeof attr === "string") { console.error('DEPRECATED: use setAttribute with an array, not a string'); return { obj: store.proxy.settings, key: attr }; } if (!Array.isArray(attr)) { return void console.error("Attribute must be string or array"); } if (attr.length === 0) { return void console.error("Attribute can't be empty"); } var obj = store.proxy.settings; attr.forEach(function (el, i) { if (i === attr.length-1) { return; } if (!obj[el]) { obj[el] = {}; } else if (typeof obj[el] !== "object") { return void console.error("Wrong attribute"); } obj = obj[el]; }); return { obj: obj, key: attr[attr.length-1] }; }; // Set the display name (username) in the proxy Store.setDisplayName = function (value, cb) { store.proxy[Constants.displayNameKey] = value; onSync(cb); }; // Reset the drive part of the userObject (from settings) Store.resetDrive = function (data, cb) { nThen(function (waitFor) { removeOwnedPads(waitFor); }).nThen(function () { store.proxy.drive = store.fo.getStructure(); onSync(cb); }); }; /** * Settings & pad attributes * data * - href (String) * - attr (Array) * - value (String) */ Store.setPadAttribute = function (data, cb) { store.userObject.setPadAttribute(data.href, data.attr, data.value, function () { onSync(cb); }); }; Store.getPadAttribute = function (data, cb) { store.userObject.getPadAttribute(data.href, data.attr, function (err, val) { if (err) { return void cb({error: err}); } cb(val); }); }; Store.setAttribute = function (data, cb) { try { var object = getAttributeObject(data.attr); object.obj[object.key] = data.value; } catch (e) { return void cb({error: e}); } onSync(cb); }; Store.getAttribute = function (data, cb) { var object; try { object = getAttributeObject(data.attr); } catch (e) { return void cb({error: e}); } cb(object.obj[object.key]); }; // Tags Store.listAllTags = function (data, cb) { var all = []; var files = Util.find(store.proxy, ['drive', 'filesData']); if (typeof(files) !== 'object') { return cb({error: 'invalid_drive'}); } Object.keys(files).forEach(function (k) { var file = files[k]; if (!Array.isArray(file.tags)) { return; } file.tags.forEach(function (tag) { if (all.indexOf(tag) === -1) { all.push(tag); } }); }); cb(all); }; // Templates Store.getTemplates = function (data, cb) { var templateFiles = store.userObject.getFiles(['template']); var res = []; templateFiles.forEach(function (f) { var data = store.userObject.getFileData(f); res.push(JSON.parse(JSON.stringify(data))); }); cb(res); }; // Pads Store.moveToTrash = function (data, cb) { var href = Hash.getRelativeHref(data.href); store.userObject.forget(href); onSync(cb); }; Store.setPadTitle = function (data, cb) { var title = data.title; var href = data.href; var p = Hash.parsePadUrl(href); var h = p.hashData; if (AppConfig.disableAnonymousStore && !store.loggedIn) { return void cb(); } var owners; if (Store.channel && Store.channel.wc && Util.base64ToHex(h.channel) === Store.channel.wc.id) { owners = Store.channel.data.owners || undefined; } var expire; if (Store.channel && Store.channel.wc && Util.base64ToHex(h.channel) === Store.channel.wc.id) { expire = +Store.channel.data.expire || undefined; } var allPads = Util.find(store.proxy, ['drive', 'filesData']) || {}; var isStronger; // If we don't find the new channel in our existing pads, we'll have to add the pads // to filesData var contains; // Update all pads that use the same channel but with a weaker hash // Edit > Edit (present) > View > View (present) for (var id in allPads) { var pad = allPads[id]; if (!pad.href) { continue; } var p2 = Hash.parsePadUrl(pad.href); var h2 = p2.hashData; // Different types, proceed to the next one // No hash data: corrupted pad? if (p.type !== p2.type || !h2) { continue; } var shouldUpdate = p.hash.replace(/\/$/, '') === p2.hash.replace(/\/$/, ''); // If the hash is different but represents the same channel, check if weaker or stronger if (!shouldUpdate && h.version === 1 && h2.version === 1 && h.channel === h2.channel) { // We had view & now we have edit, update if (h2.mode === 'view' && h.mode === 'edit') { shouldUpdate = true; } // Same mode and we had present URL, update else if (h.mode === h2.mode && h2.present) { shouldUpdate = true; } // If we're here it means we have a weaker URL: // update the date but keep the existing hash else { pad.atime = +new Date(); contains = true; continue; } } if (shouldUpdate) { contains = true; pad.atime = +new Date(); pad.title = title; pad.owners = owners; pad.expire = expire; // If the href is different, it means we have a stronger one if (href !== pad.href) { isStronger = true; } pad.href = href; } } if (isStronger) { // If we have a stronger url, remove the possible weaker from the trash. // If all of the weaker ones were in the trash, add the stronger to ROOT store.userObject.restoreHref(href); } // Add the pad if it does not exist in our drive if (!contains) { Store.addPad({ href: href, title: title, owners: owners, expire: expire, path: data.path || (store.data && store.data.initialPath) }, cb); return; } onSync(cb); }; // Filepicker app Store.getSecureFilesList = function (query, cb) { var list = {}; var hashes = []; var types = query.types; var where = query.where; var filter = query.filter || {}; var isFiltered = function (type, data) { var filtered; var fType = filter.fileType || []; if (type === 'file' && fType.length) { if (!data.fileType) { return true; } filtered = !fType.some(function (t) { return data.fileType.indexOf(t) === 0; }); } return filtered; }; store.userObject.getFiles(where).forEach(function (id) { var data = store.userObject.getFileData(id); var parsed = Hash.parsePadUrl(data.href); if ((!types || types.length === 0 || types.indexOf(parsed.type) !== -1) && hashes.indexOf(parsed.hash) === -1 && !isFiltered(parsed.type, data)) { hashes.push(parsed.hash); list[id] = data; } }); cb(list); }; Store.getPadData = function (id, cb) { cb(store.userObject.getFileData(id)); }; Store.setInitialPath = function (path) { if (!store.data) { return; } store.data.initialPath = path; }; // Messaging (manage friends from the userlist) var getMessagingCfg = function () { return { proxy: store.proxy, realtime: store.realtime, network: store.network, updateMetadata: function () { postMessage("UPDATE_METADATA"); }, pinPads: Store.pinPads, friendComplete: function (data) { postMessage("EV_FRIEND_COMPLETE", data); }, friendRequest: function (data, cb) { postMessage("Q_FRIEND_REQUEST", data, cb); }, }; }; Store.inviteFromUserlist = function (data, cb) { var messagingCfg = getMessagingCfg(); Messaging.inviteFromUserlist(messagingCfg, data, cb); }; // Messenger // Get hashes for the share button Store.getStrongerHash = function (data, cb) { var allPads = Util.find(store.proxy, ['drive', 'filesData']) || {}; // If we have a stronger version in drive, add it and add a redirect button var stronger = Hash.findStronger(data.href, allPads); if (stronger) { var parsed2 = Hash.parsePadUrl(stronger); return void cb(parsed2.hash); } cb(); }; Store.messenger = { getFriendList: function (data, cb) { store.messenger.getFriendList(function (e, keys) { cb({ error: e, data: keys, }); }); }, getMyInfo: function (data, cb) { store.messenger.getMyInfo(function (e, info) { cb({ error: e, data: info, }); }); }, getFriendInfo: function (data, cb) { store.messenger.getFriendInfo(data, function (e, info) { cb({ error: e, data: info, }); }); }, removeFriend: function (data, cb) { store.messenger.removeFriend(data, function (e, info) { cb({ error: e, data: info, }); }); }, openFriendChannel: function (data, cb) { store.messenger.openFriendChannel(data, function (e) { cb({ error: e, }); }); }, getFriendStatus: function (data, cb) { store.messenger.getStatus(data, function (e, online) { cb({ error: e, data: online, }); }); }, getMoreHistory: function (data, cb) { store.messenger.getMoreHistory(data.curvePublic, data.sig, data.count, function (e, history) { cb({ error: e, data: history, }); }); }, sendMessage: function (data, cb) { store.messenger.sendMessage(data.curvePublic, data.content, function (e) { cb({ error: e, }); }); }, setChannelHead: function (data, cb) { store.messenger.setChannelHead(data.curvePublic, data.sig, function (e) { cb({ error: e }); }); } }; ////////////////////////////////////////////////////////////////// /////////////////////// PAD ////////////////////////////////////// ////////////////////////////////////////////////////////////////// // TODO with sharedworker // channel will be an object storing the webchannel associated to each browser tab var channel = Store.channel = { queue: [], data: {} }; Store.joinPad = function (data, cb) { var conf = { onReady: function (padData) { channel.data = padData || {}; postMessage("PAD_READY"); }, // post EV_PAD_READY onMessage: function (m) { postMessage("PAD_MESSAGE", m); }, // post EV_PAD_MESSAGE onJoin: function (m) { postMessage("PAD_JOIN", m); }, // post EV_PAD_JOIN onLeave: function (m) { postMessage("PAD_LEAVE", m); }, // post EV_PAD_LEAVE onDisconnect: function () { postMessage("PAD_DISCONNECT"); }, // post EV_PAD_DISCONNECT onError: function (err) { postMessage("PAD_ERROR", err); }, // post EV_PAD_ERROR channel: data.channel, validateKey: data.validateKey, owners: data.owners, password: data.password, expire: data.expire, network: store.network, readOnly: data.readOnly, onConnect: function (wc, sendMessage) { channel.sendMessage = sendMessage; channel.wc = wc; channel.queue.forEach(function (data) { sendMessage(data.message); }); cb({ myID: wc.myID, id: wc.id, members: wc.members }); } }; CpNfWorker.start(conf); }; Store.sendPadMsg = function (data, cb) { if (!channel.wc) { channel.queue.push(data); } channel.sendMessage(data, cb); }; // TODO // GET_FULL_HISTORY from sframe-common-outer Store.getFullHistory = function (data, cb) { var network = store.network; var hkn = network.historyKeeper; //var crypto = Crypto.createEncryptor(data.keys); // Get the history messages and send them to the iframe var parse = function (msg) { try { return JSON.parse(msg); } catch (e) { return null; } }; var msgs = []; var onMsg = function (msg) { var parsed = parse(msg); if (parsed[0] === 'FULL_HISTORY_END') { cb(msgs); return; } if (parsed[0] !== 'FULL_HISTORY') { return; } if (parsed[1] && parsed[1].validateKey) { // First message return; } if (parsed[1][3] !== data.channel) { return; } msg = parsed[1][4]; if (msg) { msg = msg.replace(/^cp\|/, ''); //var decryptedMsg = crypto.decrypt(msg, true); msgs.push(msg); } }; network.on('message', onMsg); network.sendto(hkn, JSON.stringify(['GET_FULL_HISTORY', data.channel, data.validateKey])); }; // TODO with sharedworker // when the tab is closed, leave the pad // Drive Store.userObjectCommand = function (cmdData, cb) { if (!cmdData || !cmdData.cmd) { return; } var data = cmdData.data; switch (cmdData.cmd) { case 'move': store.userObject.move(data.paths, data.newPath, cb); break; case 'restore': store.userObject.restore(data.path, cb); break; case 'addFolder': store.userObject.addFolder(data.path, data.name, cb); break; case 'delete': store.userObject.delete(data.paths, cb, data.nocheck, data.isOwnPadRemoved); break; case 'emptyTrash': store.userObject.emptyTrash(cb); break; case 'rename': store.userObject.rename(data.path, data.newName, cb); break; default: cb(); } }; ////////////////////////////////////////////////////////////////// /////////////////////// Init ///////////////////////////////////// ////////////////////////////////////////////////////////////////// var onReady = function (returned, cb) { var proxy = store.proxy; var userObject = store.userObject = UserObject.init(proxy.drive, { pinPads: Store.pinPads, unpinPads: Store.unpinPads, removeOwnedChannel: Store.removeOwnedChannel, edPublic: store.proxy.edPublic, loggedIn: store.loggedIn, log: function (msg) { postMessage("DRIVE_LOG", msg); } }); var todo = function () { userObject.fixFiles(); Migrate(proxy); var requestLogin = function () { postMessage("REQUEST_LOGIN"); }; if (store.loggedIn) { /* This isn't truly secure, since anyone who can read the user's object can set their local loginToken to match that in the object. However, it exposes a UI that will work most of the time. */ // every user object should have a persistent, random number if (typeof(proxy.loginToken) !== 'number') { proxy[Constants.tokenKey] = Math.floor(Math.random()*Number.MAX_SAFE_INTEGER); } returned[Constants.tokenKey] = proxy[Constants.tokenKey]; if (store.data.localToken && store.data.localToken !== proxy[Constants.tokenKey]) { // the local number doesn't match that in // the user object, request that they reauthenticate. return void requestLogin(); } } if (!proxy.settings || !proxy.settings.general || typeof(proxy.settings.general.allowUserFeedback) !== 'boolean') { proxy.settings = proxy.settings || {}; proxy.settings.general = proxy.settings.general || {}; proxy.settings.general.allowUserFeedback = true; } returned.feedback = proxy.settings.general.allowUserFeedback; if (typeof(cb) === 'function') { cb(returned); } if (typeof(proxy.uid) !== 'string' || proxy.uid.length !== 32) { // even anonymous users should have a persistent, unique-ish id console.log('generating a persistent identifier'); proxy.uid = Hash.createChannelId(); } // if the user is logged in, but does not have signing keys... if (store.loggedIn && (!Store.hasSigningKeys() || !Store.hasCurveKeys())) { return void requestLogin(); } proxy.on('change', [Constants.displayNameKey], function (o, n) { if (typeof(n) !== "string") { return; } postMessage("UPDATE_METADATA"); }); proxy.on('change', ['profile'], function () { // Trigger userlist update when the avatar has changed postMessage("UPDATE_METADATA"); }); proxy.on('change', ['friends'], function () { // Trigger userlist update when the friendlist has changed postMessage("UPDATE_METADATA"); }); proxy.on('change', ['settings'], function () { postMessage("UPDATE_METADATA"); }); proxy.on('change', [Constants.tokenKey], function () { postMessage("UPDATE_TOKEN", { token: proxy[Constants.tokenKey] }); }); }; userObject.migrate(todo); }; var connect = function (data, cb) { var hash = data.userHash || data.anonHash || Hash.createRandomHash(); storeHash = hash; if (!hash) { throw new Error('[Store.init] Unable to find or create a drive hash. Aborting...'); } var secret = Hash.getSecrets('drive', hash); var listmapConfig = { data: {}, websocketURL: NetConfig.getWebsocketURL(), channel: secret.channel, readOnly: false, validateKey: secret.keys.validateKey || undefined, crypto: Crypto.createEncryptor(secret.keys), userName: 'fs', logLevel: 1, ChainPad: ChainPad, classic: true, }; var rt = window.rt = Listmap.create(listmapConfig); store.proxy = rt.proxy; store.loggedIn = typeof(data.userHash) !== "undefined"; var returned = {}; rt.proxy.on('create', function (info) { store.realtime = info.realtime; store.network = info.network; if (!data.userHash) { returned.anonHash = Hash.getEditHashFromKeys(info.channel, secret.keys); } }).on('ready', function () { if (store.userObject) { return; } // the store is already ready, it is a reconnection if (!rt.proxy.drive || typeof(rt.proxy.drive) !== 'object') { rt.proxy.drive = {}; } var drive = rt.proxy.drive; // Creating a new anon drive: import anon pads from localStorage if ((!drive[Constants.oldStorageKey] || !Array.isArray(drive[Constants.oldStorageKey])) && !drive['filesData']) { drive[Constants.oldStorageKey] = []; } // Drive already exist: return the existing drive, don't load data from legacy store onReady(returned, cb); }) .on('change', ['drive', 'migrate'], function () { var path = arguments[2]; var value = arguments[1]; if (path[0] === 'drive' && path[1] === "migrate" && value === 1) { rt.network.disconnect(); rt.realtime.abort(); postMessage('NETWORK_DISCONNECT'); } }); rt.proxy.on('disconnect', function () { postMessage('NETWORK_DISCONNECT'); }); rt.proxy.on('reconnect', function (info) { postMessage('NETWORK_RECONNECT', {myId: info.myId}); }); }; /** * Data: * - userHash or anonHash * Todo in cb * - LocalStore.setFSHash if needed * - sessionStorage.User_Hash * - stuff with tokenKey * Event to outer * - requestLogin */ var initialized = false; Store.init = function (data, callback) { if (initialized) { return void callback({ state: 'ALREADY_INIT', returned: store.returned }); } initialized = true; postMessage = function (cmd, d, cb) { setTimeout(function () { data.query(cmd, d, cb); // TODO temporary, will be replaced by webworker channel }); }; store.data = data; connect(data, function (ret) { if (Object.keys(store.proxy).length === 1) { Feedback.send("FIRST_APP_USE", true); } store.returned = ret; callback(ret); var messagingCfg = getMessagingCfg(); Messaging.addDirectMessageHandler(messagingCfg); // Send events whenever there is a change or a removal in the drive if (data.driveEvents) { store.proxy.on('change', [], function (o, n, p) { postMessage('DRIVE_CHANGE', { old: o, new: n, path: p }); }); store.proxy.on('remove', [], function (o, p) { postMessage('DRIVE_REMOVE', { old: o, path: p }); }); } if (data.messenger) { var messenger = store.messenger = Messenger.messenger(store); messenger.on('message', function (message) { postMessage('CONTACTS_MESSAGE', message); }); messenger.on('join', function (curvePublic, channel) { postMessage('CONTACTS_JOIN', { curvePublic: curvePublic, channel: channel, }); }); messenger.on('leave', function (curvePublic, channel) { postMessage('CONTACTS_LEAVE', { curvePublic: curvePublic, channel: channel, }); }); messenger.on('update', function (info, curvePublic) { postMessage('CONTACTS_UPDATE', { curvePublic: curvePublic, info: info, }); }); messenger.on('friend', function (curvePublic) { postMessage('CONTACTS_FRIEND', { curvePublic: curvePublic, }); }); messenger.on('unfriend', function (curvePublic) { postMessage('CONTACTS_UNFRIEND', { curvePublic: curvePublic, }); }); } }); }; Store.disconnect = function () { if (!store.network) { return; } store.network.disconnect(); }; return Store; });