'use strict'; angular.module('copayApp.services') .factory('storageService', function(appConfigService, logHeader, fileStorageService, localStorageService, sjcl, $log, lodash, platformInfo, $timeout) { var root = {}; var storage; // File storage is not supported for writing according to // https://github.com/apache/cordova-plugin-file/#supported-platforms var shouldUseFileStorage = platformInfo.isCordova && !platformInfo.isWP; if (shouldUseFileStorage) { $log.debug('Using: FileStorage'); storage = fileStorageService; } else { $log.debug('Using: LocalStorage'); storage = localStorageService; } var getUUID = function(cb) { // TO SIMULATE MOBILE //return cb('hola'); if (!window || !window.plugins || !window.plugins.uniqueDeviceID) return cb(null); window.plugins.uniqueDeviceID.get( function(uuid) { return cb(uuid); }, cb); }; // This is only used in Copay, we used to encrypt profile // using device's UUID. var decryptOnMobile = function(text, cb) { var json; try { json = JSON.parse(text); } catch (e) { $log.warn('Could not open profile:' + text); var i = text.lastIndexOf('}{'); if (i > 0) { text = text.substr(i + 1); $log.warn('trying last part only:' + text); try { json = JSON.parse(text); $log.warn('Worked... saving.'); storage.set('profile', text, function() {}); } catch (e) { $log.warn('Could not open profile (2nd try):' + e); }; }; }; if (!json) return cb('Could not access storage') if (!json.iter || !json.ct) { $log.debug('Profile is not encrypted'); return cb(null, text); } $log.debug('Profile is encrypted'); getUUID(function(uuid) { $log.debug('Device UUID:' + uuid); if (!uuid) return cb('Could not decrypt storage: could not get device ID'); try { text = sjcl.decrypt(uuid, text); $log.info('Migrating to unencrypted profile'); return storage.set('profile', text, function(err) { return cb(err, text); }); } catch (e) { $log.warn('Decrypt error: ', e); return cb('Could not decrypt storage: device ID mismatch'); }; return cb(null, text); }); }; // This is only use in Copay, for very old instalations // in which we use to use localStorage instead of fileStorage root.tryToMigrate = function(cb) { if (!shouldUseFileStorage) return cb(); localStorageService.get('profile', function(err, str) { if (err) return cb(err); if (!str) return cb(); $log.info('Starting Migration profile to File storage...'); fileStorageService.create('profile', str, function(err) { if (err) cb(err); $log.info('Profile Migrated successfully'); localStorageService.get('config', function(err, c) { if (err) return cb(err); if (!c) return root.getProfile(cb); fileStorageService.create('config', c, function(err) { if (err) { $log.info('Error migrating config: ignoring', err); return root.getProfile(cb); } $log.info('Config Migrated successfully'); return root.getProfile(cb); }); }); }); }); }; root.storeNewProfile = function(profile, cb) { root.storeProfile(profile, cb); }; root.storeProfile = function(profile, cb) { var profileString = profile.toObj(); storage.set('profile', profileString, cb); }; /** * @callback getProfileCallback * @param {Error} error - falsy if profile not found. * @param {Profile} profile - falsy if error or profile not found. */ /** * @param {Error} error * @param {String} profileStr - containing the profile * @param {getProfileCallback} cb */ function _onOldProfileRetrieved(error, profileStr, cb) { if (error) { return cb(error, null); } if (!profileStr) { // No profiles found. No errors either. return cb(null, null); } decryptOnMobile(profileStr, function(decryptErr, decryptedStr) { if (decryptErr) return cb(decryptErr, null); var profile; try { profile = Profile.fromString(decryptedStr); } catch (e) { $log.debug('Could not read profile:', e); return(new Error('Could not read profile.'), null); } cb(null, profile) }); } /** * * @param {Profile} oldProfile * @param {Profile} secureProfile - may be falsy if no secure profile found. * @param {getProfileCallback} cb */ function _migrateProfiles(oldProfile, secureProfile, cb) { var newProfile; if (secureProfile) { secureProfile.merge(oldProfile); newProfile = secureProfile; } else { newProfile = oldProfile; newProfile.setAppVersion(appConfigService.version); } root.storeNewProfile(newProfile, function(storeErr) { if (storeErr) { cb(storeErr, null); return; } storage.remove('profile', function(removeErr){ if (removeErr) { cb(removeErr, null); return; } cb(null, newProfile); }); }); }; /** * * @param {getProfileCallback} cb */ root.getProfile = function(cb) { storage.get('profile', function(getErr, getStr) { if (getErr) { cb(getErr, null); } else { profile = Profile.fromString(getStr); cb(null, profile); } }); }; root.setFeedbackInfo = function(feedbackValues, cb) { storage.set('feedback', feedbackValues, cb); }; root.getFeedbackInfo = function(cb) { storage.get('feedback', cb); }; root.storeFocusedWalletId = function(id, cb) { storage.set('focusedWalletId', id || '', cb); }; root.getFocusedWalletId = function(cb) { storage.get('focusedWalletId', cb); }; root.getLastAddress = function(walletId, cb) { storage.get('lastAddress-' + walletId, cb); }; root.storeLastAddress = function(walletId, address, cb) { storage.set('lastAddress-' + walletId, address, cb); }; root.clearLastAddress = function(walletId, cb) { storage.remove('lastAddress-' + walletId, cb); }; root.setBackupFlag = function(walletId, cb) { storage.set('backup-' + walletId, Date.now(), cb); }; root.getBackupFlag = function(walletId, cb) { storage.get('backup-' + walletId, cb); }; root.clearBackupFlag = function(walletId, cb) { storage.remove('backup-' + walletId, cb); }; root.setCleanAndScanAddresses = function(walletId, cb) { storage.set('CleanAndScanAddresses', walletId, cb); }; root.getCleanAndScanAddresses = function(cb) { storage.get('CleanAndScanAddresses', cb); }; root.removeCleanAndScanAddresses = function(cb) { storage.remove('CleanAndScanAddresses', cb); }; root.getConfig = function(cb) { storage.get('config', cb); }; root.storeConfig = function(val, cb) { $log.debug('Storing Preferences', val); storage.set('config', val, cb); }; root.clearConfig = function(cb) { storage.remove('config', cb); }; root.getHomeTipAccepted = function(cb) { storage.get('homeTip', cb); }; root.setHomeTipAccepted = function(val, cb) { storage.set('homeTip', val, cb); }; root.setHideBalanceFlag = function(walletId, val, cb) { storage.set('hideBalance-' + walletId, val, cb); }; root.getHideBalanceFlag = function(walletId, cb) { storage.get('hideBalance-' + walletId, cb); }; //for compatibility root.getCopayDisclaimerFlag = function(cb) { storage.get('agreeDisclaimer', cb); }; root.setRemotePrefsStoredFlag = function(cb) { storage.set('remotePrefStored', true, cb); }; root.getRemotePrefsStoredFlag = function(cb) { storage.get('remotePrefStored', cb); }; root.setGlideraToken = function(network, token, cb) { storage.set('glideraToken-' + network, token, cb); }; root.getGlideraToken = function(network, cb) { storage.get('glideraToken-' + network, cb); }; root.removeGlideraToken = function(network, cb) { storage.remove('glideraToken-' + network, cb); }; root.setGlideraPermissions = function(network, p, cb) { storage.set('glideraPermissions-' + network, p, cb); }; root.getGlideraPermissions = function(network, cb) { storage.get('glideraPermissions-' + network, cb); }; root.removeGlideraPermissions = function(network, cb) { storage.remove('glideraPermissions-' + network, cb); }; root.setGlideraStatus = function(network, status, cb) { storage.set('glideraStatus-' + network, status, cb); }; root.getGlideraStatus = function(network, cb) { storage.get('glideraStatus-' + network, cb); }; root.removeGlideraStatus = function(network, cb) { storage.remove('glideraStatus-' + network, cb); }; root.setGlideraTxs = function(network, txs, cb) { storage.set('glideraTxs-' + network, txs, cb); }; root.getGlideraTxs = function(network, cb) { storage.get('glideraTxs-' + network, cb); }; root.removeGlideraTxs = function(network, cb) { storage.remove('glideraTxs-' + network, cb); }; root.setCoinbaseRefreshToken = function(network, token, cb) { storage.set('coinbaseRefreshToken-' + network, token, cb); }; root.getCoinbaseRefreshToken = function(network, cb) { storage.get('coinbaseRefreshToken-' + network, cb); }; root.removeCoinbaseRefreshToken = function(network, cb) { storage.remove('coinbaseRefreshToken-' + network, cb); }; root.setCoinbaseToken = function(network, token, cb) { storage.set('coinbaseToken-' + network, token, cb); }; root.getCoinbaseToken = function(network, cb) { storage.get('coinbaseToken-' + network, cb); }; root.removeCoinbaseToken = function(network, cb) { storage.remove('coinbaseToken-' + network, cb); }; root.setAddressbook = function(network, addressbook, cb) { storage.set('addressbook-' + network, addressbook, cb); }; root.getAddressbook = function(network, cb) { storage.get('addressbook-' + network, cb); }; root.removeAddressbook = function(network, cb) { storage.remove('addressbook-' + network, cb); }; root.setLastCurrencyUsed = function(lastCurrencyUsed, cb) { storage.set('lastCurrencyUsed', lastCurrencyUsed, cb) }; root.getLastCurrencyUsed = function(cb) { storage.get('lastCurrencyUsed', cb) }; root.checkQuota = function() { var block = ''; // 50MB for (var i = 0; i < 1024 * 1024; ++i) { block += '12345678901234567890123456789012345678901234567890'; } storage.set('test', block, function(err) { $log.error('CheckQuota Return:' + err); }); }; root.setTxHistory = function(txs, walletId, cb) { try { storage.set('txsHistory-' + walletId, txs, cb); } catch (e) { $log.error('Error saving tx History. Size:' + txs.length); $log.error(e); return cb(e); } } root.getTxHistory = function(walletId, cb) { storage.get('txsHistory-' + walletId, cb); } root.removeTxHistory = function(walletId, cb) { storage.remove('txsHistory-' + walletId, cb); } root.setCoinbaseTxs = function(network, ctx, cb) { storage.set('coinbaseTxs-' + network, ctx, cb); }; root.getCoinbaseTxs = function(network, cb) { storage.get('coinbaseTxs-' + network, cb); }; root.removeCoinbaseTxs = function(network, cb) { storage.remove('coinbaseTxs-' + network, cb); }; root.setBalanceCache = function(cardId, data, cb) { storage.set('balanceCache-' + cardId, data, cb); }; root.getBalanceCache = function(cardId, cb) { storage.get('balanceCache-' + cardId, cb); }; root.removeBalanceCache = function(cardId, cb) { storage.remove('balanceCache-' + cardId, cb); }; // cards: [ // eid: card id // id: card id // lastFourDigits: card number // token: card token // ] root.setBitpayDebitCards = function(network, email, cards, cb) { root.getBitpayAccounts(network, function(err, allAccounts) { if (err) return cb(err); if (!allAccounts[email]) { return cb('Cannot set cards for unknown account ' + email); } allAccounts[email].cards = cards; storage.set('bitpayAccounts-v2-' + network, allAccounts, cb); }); }; // cb(err, cards) // cards: [ // eid: card id // id: card id // lastFourDigits: card number // token: card token // email: account email // ] root.getBitpayDebitCards = function(network, cb) { root.getBitpayAccounts(network, function(err, allAccounts) { if (err) return cb(err); var allCards = []; lodash.each(allAccounts, function(account, email) { if (account.cards) { // Add account's email to each card var cards = lodash.clone(account.cards); lodash.each(cards, function(x) { x.email = email; }); allCards = allCards.concat(cards); } }); return cb(null, allCards); }); }; root.removeBitpayDebitCard = function(network, cardEid, cb) { root.getBitpayAccounts(network, function(err, allAccounts) { lodash.each(allAccounts, function(account) { account.cards = lodash.reject(account.cards, { 'eid': cardEid }); }); storage.set('bitpayAccounts-v2-' + network, allAccounts, cb); }); }; // cb(err, accounts) // accounts: { // email_1: { // token: account token // cards: { // // } // } // ... // email_n: { // token: account token // cards: { // // } // } // } // root.getBitpayAccounts = function(network, cb) { storage.get('bitpayAccounts-v2-' + network, function(err, allAccountsStr) { if (err) return cb(err); if (!allAccountsStr) return cb(null, {}); var allAccounts = {}; try { allAccounts = JSON.parse(allAccountsStr); } catch (e) { $log.error('Bad storage value for bitpayAccount-v2' + allAccountsStr) return cb(null, {}); }; var anyMigration; lodash.each(allAccounts, function(account, email) { // Migrate old `'bitpayApi-' + network` key, if exists if (!account.token && account['bitpayApi-' + network].token) { $log.info('Migrating all bitpayApi-network branch'); account.token = account['bitpayApi-' + network].token; account.cards = lodash.clone(account['bitpayApi-' + network].cards); if (!account.cards) { account.cards = lodash.clone(account['bitpayDebitCards-' + network]); } delete account['bitpayDebitCards-' + network]; delete account['bitpayApi-' + network]; anyMigration = true; } }); if (anyMigration) { storage.set('bitpayAccounts-v2-' + network, allAccounts, function() { return cb(err, allAccounts); }); } else return cb(err, allAccounts); }); }; // data: { // email: account email // token: account token // familyName: account family (last) name // givenName: account given (first) name // } root.setBitpayAccount = function(network, data, cb) { if (!lodash.isObject(data) || !data.email || !data.token) return cb('No account to set'); root.getBitpayAccounts(network, function(err, allAccounts) { if (err) return cb(err); allAccounts = allAccounts || {}; var account = allAccounts[data.email] || {}; account.token = data.token; account.familyName = data.familyName; account.givenName = data.givenName; allAccounts[data.email] = account; $log.info('Storing BitPay accounts with new account:' + data.email); storage.set('bitpayAccounts-v2-' + network, allAccounts, cb); }); }; // account: { // email: account email // apiContext: the context needed for making future api calls // cards: an array of cards // } root.removeBitpayAccount = function(network, account, cb) { if (lodash.isString(account)) { account = JSON.parse(account); } account = account || {}; if (lodash.isEmpty(account)) return cb('No account to remove'); storage.get('bitpayAccounts-v2-' + network, function(err, bitpayAccounts) { if (err) cb(err); if (lodash.isString(bitpayAccounts)) { bitpayAccounts = JSON.parse(bitpayAccounts); } bitpayAccounts = bitpayAccounts || {}; delete bitpayAccounts[account.email]; storage.set('bitpayAccounts-v2-' + network, JSON.stringify(bitpayAccounts), cb); }); }; root.setAppIdentity = function(network, data, cb) { storage.set('appIdentity-' + network, data, cb); }; root.getAppIdentity = function(network, cb) { storage.get('appIdentity-' + network, function(err, data) { if (err) return cb(err); cb(err, JSON.parse(data || '{}')); }); }; root.removeAppIdentity = function(network, cb) { storage.remove('appIdentity-' + network, cb); }; root.removeAllWalletData = function(walletId, cb) { root.clearLastAddress(walletId, function(err) { if (err) return cb(err); root.removeTxHistory(walletId, function(err) { if (err) return cb(err); root.clearBackupFlag(walletId, function(err) { return cb(err); }); }); }); }; root.setAmazonGiftCards = function(network, gcs, cb) { storage.set('amazonGiftCards-' + network, gcs, cb); }; root.getAmazonGiftCards = function(network, cb) { storage.get('amazonGiftCards-' + network, cb); }; root.removeAmazonGiftCards = function(network, cb) { storage.remove('amazonGiftCards-' + network, cb); }; root.setTxConfirmNotification = function(txid, val, cb) { storage.set('txConfirmNotif-' + txid, val, cb); }; root.getTxConfirmNotification = function(txid, cb) { storage.get('txConfirmNotif-' + txid, cb); }; root.removeTxConfirmNotification = function(txid, cb) { storage.remove('txConfirmNotif-' + txid, cb); }; root.setMercadoLibreGiftCards = function(network, gcs, cb) { storage.set('mercadoLibreGiftCards-' + network, gcs, cb); }; root.getMercadoLibreGiftCards = function(network, cb) { storage.get('mercadoLibreGiftCards-' + network, cb); }; root.removeMercadoLibreGiftCards = function(network, cb) { storage.remove('MercadoLibreGiftCards-' + network, cb); }; root.setReceivedTransactions = function(walletId, txIds, cb) { storage.set('receivedTxs-' + walletId, txIds, cb); } root.getReceivedTransactions = function(walletId, cb) { storage.get('receivedTxs-' + walletId, cb); } root.removeReceivedTransactions = function(walletId, cb) { storage.remove('receivedTxs-' + walletId, cb); } root.checkIfFlagIsSet = function(key) { return new Promise(function(resolve, reject) { storage.get(key, function(i, value) { if (value == null) { resolve(false); } else { resolve(JSON.parse(value).initialized); } }); }); } return root; });