define([ 'jquery', '/bower_components/chainpad-listmap/chainpad-listmap.js', '/bower_components/chainpad-crypto/crypto.js?v=0.1.5', '/bower_components/textpatcher/TextPatcher.amd.js', '/common/userObject.js', '/common/common-interface.js', '/common/common-hash.js', '/common/common-constants.js', '/common/migrate-user-object.js', ], function ($, Listmap, Crypto, TextPatcher, FO, UI, Hash, Constants, Migrate) { /* This module uses localStorage, which is synchronous, but exposes an asyncronous API. This is so that we can substitute other storage methods. To override these methods, create another file at: /customize/storage.js */ var Store = {}; var store; var initStore = function (filesOp, storeObj, exp) { var ret = {}; var safeSet = function (key, val) { storeObj[key] = val; }; // Store uses nodebacks... ret.set = function (key, val, cb) { safeSet(key, val); cb(); }; // implement in alternative store ret.setBatch = function (map, cb) { Object.keys(map).forEach(function (key) { safeSet(key, map[key]); }); cb(void 0, map); }; ret.setDrive = function (key, val, cb) { storeObj.drive[key] = val; cb(); }; var safeGet = function (key) { return storeObj[key]; }; ret.get = function (key, cb) { cb(void 0, safeGet(key)); }; // implement in alternative store ret.getBatch = function (keys, cb) { var res = {}; keys.forEach(function (key) { res[key] = safeGet(key); }); cb(void 0, res); }; var getAttributeObject = function (attr) { if (typeof attr === "string") { console.error('DEPRECATED: use setAttribute with an array, not a string'); return { obj: storeObj.settings, key: attr }; } if (!Array.isArray(attr)) { throw new Error("Attribute must be string or array"); } if (attr.length === 0) { throw new Error("Attribute can't be empty"); } var obj = storeObj.settings; attr.forEach(function (el, i) { if (i === attr.length-1) { return; } if (!obj[el]) { obj[el] = {}; } else if (typeof obj[el] !== "object") { throw new Error("Wrong attribute"); } obj = obj[el]; }); return { obj: obj, key: attr[attr.length-1] }; }; ret.setAttribute = function (attr, value, cb) { try { var object = getAttributeObject(attr); object.obj[object.key] = value; } catch (e) { return void cb(e); } cb(); }; ret.getAttribute = function (attr, cb) { var object; try { object = getAttributeObject(attr); } catch (e) { return void cb(e); } cb(null, object.obj[object.key]); }; ret.setPadAttribute = filesOp.setPadAttribute; ret.getPadAttribute = filesOp.getPadAttribute; ret.getIdFromHref = filesOp.getIdFromHref; ret.getDrive = function (key, cb) { cb(void 0, storeObj.drive[key]); }; var safeRemove = function (key) { delete storeObj[key]; }; ret.remove = function (key, cb) { safeRemove(key); cb(); }; // implement in alternative store ret.removeBatch = function (keys, cb) { keys.forEach(function (key) { safeRemove(key); }); cb(); }; ret.keys = function (cb) { cb(void 0, Object.keys(storeObj)); }; ret.removeData = filesOp.removeData; ret.pushData = filesOp.pushData; ret.addPad = filesOp.add; ret.forgetPad = function (href, cb) { filesOp.forget(href); cb(); }; ret.listTemplates = function () { var templateFiles = filesOp.getFiles(['template']); var res = []; templateFiles.forEach(function (f) { var data = filesOp.getFileData(f); res.push(JSON.parse(JSON.stringify(data))); }); return res; }; ret.getProxy = function () { return exp; }; ret.getLoginName = function () { return storeObj.login_name; }; ret.repairDrive = function () { filesOp.fixFiles(); }; ret.getEmptyObject = function () { return filesOp.getStructure(); }; ret.replace = filesOp.replace; ret.restoreHref = filesOp.restoreHref; ret.changeHandlers = []; ret.change = function () {}; ret.getProfile = function () { return storeObj.profile; }; return ret; }; var tryParsing = function (x) { try { return JSON.parse(x); } catch (e) { console.error(e); return null; } }; var onReady = function (f, proxy, Cryptpad, exp) { var fo = exp.fo = FO.init(proxy.drive, { Cryptpad: Cryptpad, loggedIn: Cryptpad.isLoggedIn() }); var todo = function () { fo.fixFiles(); Migrate(proxy, Cryptpad); store = initStore(fo, proxy, exp); if (typeof(f) === 'function') { f(void 0, store); } //storeObj = proxy; var requestLogin = function () { // log out so that you don't go into an endless loop... Cryptpad.logout(); // redirect them to log in, and come back when they're done. sessionStorage.redirectTo = window.location.href; window.location.href = '/login/'; }; var tokenKey = 'loginToken'; if (Cryptpad.isLoggedIn()) { /* 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[tokenKey] = Math.floor(Math.random()*Number.MAX_SAFE_INTEGER); } // copy User_hash into sessionStorage because cross-domain iframes // on safari replaces localStorage with sessionStorage or something if (sessionStorage) { sessionStorage.setItem('User_hash', localStorage.getItem('User_hash')); } var localToken = tryParsing(localStorage.getItem(tokenKey)); if (localToken === null) { // if that number hasn't been set to localStorage, do so. localStorage.setItem(tokenKey, proxy.loginToken); } else if (localToken !== proxy[tokenKey]) { // if it has been, and 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; } 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 (Cryptpad.isLoggedIn() && (!Cryptpad.hasSigningKeys(proxy) || !Cryptpad.hasCurveKeys(proxy))) { return void requestLogin(); } proxy.on('change', [Constants.displayNameKey], function (o, n) { if (typeof(n) !== "string") { return; } Cryptpad.changeDisplayName(n); }); proxy.on('change', ['profile'], function () { // Trigger userlist update when the avatar has changed Cryptpad.changeDisplayName(proxy[Constants.displayNameKey]); }); proxy.on('change', ['friends'], function () { // Trigger userlist update when the avatar has changed Cryptpad.changeDisplayName(proxy[Constants.displayNameKey]); }); proxy.on('change', [tokenKey], function () { var localToken = tryParsing(localStorage.getItem(tokenKey)); if (localToken !== proxy[tokenKey]) { return void requestLogin(); } }); }; fo.migrate(todo); }; var initialized = false; var init = function (f, Cryptpad) { if (!Cryptpad || initialized) { return; } initialized = true; var hash = Cryptpad.getUserHash() || localStorage.FS_hash || Hash.createRandomHash(); 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: Cryptpad.getWebsocketURL(), channel: secret.channel, readOnly: false, validateKey: secret.keys.validateKey || undefined, crypto: Crypto.createEncryptor(secret.keys), userName: 'fs', logLevel: 1, }; var exp = {}; var rt = window.rt = Listmap.create(listmapConfig); exp.realtime = rt.realtime; exp.proxy = rt.proxy; rt.proxy.on('create', function (info) { exp.info = info; if (!Cryptpad.getUserHash()) { localStorage.FS_hash = Hash.getEditHashFromKeys(info.channel, secret.keys); } }).on('ready', function () { if (store) { 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] = []; onReady(f, rt.proxy, Cryptpad, exp); return; } // Drive already exist: return the existing drive, don't load data from legacy store onReady(f, rt.proxy, Cryptpad, exp); }) .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(); UI.alert(Cryptpad.Messages.fs_migration, null, true); } }); }; Store.ready = function (f, Cryptpad) { if (store) { // Store.ready probably called twice, store already ready if (typeof(f) === 'function') { f(void 0, store); } } else { init(f, Cryptpad); } }; return Store; });