1
0
mirror of https://github.com/bitwarden/browser.git synced 2024-10-19 07:35:48 +02:00
bitwarden-browser/src/services/loginService.js

640 lines
20 KiB
JavaScript
Raw Normal View History

2017-10-14 20:27:14 +02:00
function LoginService(cryptoService, userService, apiService, settingsService, utilsService, constantsService) {
2017-01-04 00:40:07 +01:00
this.cryptoService = cryptoService;
this.userService = userService;
this.apiService = apiService;
this.settingsService = settingsService;
this.utilsService = utilsService;
2017-10-14 20:27:14 +02:00
this.constantsService = constantsService;
this.decryptedCipherCache = null;
this.localDataKey = 'sitesLocalData';
2017-10-14 20:27:14 +02:00
this.neverDomainsKey = 'neverDomains';
2017-01-04 00:40:07 +01:00
initLoginService();
2017-07-14 21:34:05 +02:00
}
2017-01-04 00:40:07 +01:00
function initLoginService() {
LoginService.prototype.clearCache = function () {
this.decryptedCipherCache = null;
2017-01-04 00:40:07 +01:00
};
LoginService.prototype.encrypt = function (login) {
2017-06-27 05:55:51 +02:00
var self = this;
2017-01-04 00:40:07 +01:00
var model = {
id: login.id,
folderId: login.folderId,
2017-04-24 19:58:32 +02:00
favorite: login.favorite,
2017-10-14 04:34:36 +02:00
organizationId: login.organizationId,
type: login.type
2017-01-04 00:40:07 +01:00
};
2017-06-27 05:55:51 +02:00
return self.cryptoService.getOrgKey(login.organizationId).then(function (key) {
2017-10-14 04:34:36 +02:00
return Q.all([
encryptObjProperty(login, model, {
2017-10-14 20:27:14 +02:00
name: null,
notes: null
2017-10-14 04:34:36 +02:00
}, key, self),
2017-10-16 16:09:17 +02:00
encryptCipherData(login, model, key, self),
2017-10-14 04:34:36 +02:00
self.encryptFields(login.fields, key).then(function (fields) {
model.fields = fields;
})
]);
}).then(function () {
2017-09-22 04:45:24 +02:00
return model;
});
};
2017-10-16 16:09:17 +02:00
function encryptCipherData(cipher, model, key, self) {
switch (cipher.type) {
case self.constantsService.cipherType.login:
model.login = {};
return encryptObjProperty(cipher.login, model.login, {
uri: null,
username: null,
password: null,
totp: null
}, key, self);
case self.constantsService.cipherType.secureNote:
model.secureNote = {
type: cipher.secureNote.type
};
return Q();
case self.constantsService.cipherType.card:
model.card = {};
return encryptObjProperty(cipher.card, model.card, {
cardholderName: null,
brand: null,
number: null,
expMonth: null,
expYear: null,
code: null
}, key, self);
case self.constantsService.cipherType.identity:
model.identity = {};
return encryptObjProperty(cipher.identity, model.identity, {
title: null,
firstName: null,
middleName: null,
lastName: null,
address1: null,
address2: null,
address3: null,
city: null,
state: null,
postalCode: null,
country: null,
company: null,
email: null,
phone: null,
ssn: null,
username: null,
passportNumber: null,
licenseNumber: null
}, key, self);
default:
throw 'Unknown type.';
}
}
2017-09-22 04:45:24 +02:00
LoginService.prototype.encryptFields = function (fields, key) {
var self = this;
if (!fields || !fields.length) {
2017-10-16 16:09:17 +02:00
return Q(null);
2017-09-22 04:45:24 +02:00
}
var encFields = [];
return fields.reduce(function (promise, field) {
return promise.then(function () {
return self.encryptField(field, key);
}).then(function (encField) {
encFields.push(encField);
});
}, Q()).then(function () {
return encFields;
});
};
LoginService.prototype.encryptField = function (field, key) {
var self = this;
var model = {
type: field.type
};
2017-10-14 04:34:36 +02:00
return encryptObjProperty(field, model, {
2017-10-14 20:27:14 +02:00
name: null,
value: null
2017-10-14 04:34:36 +02:00
}, key, self).then(function () {
2017-01-04 00:40:07 +01:00
return model;
});
};
2017-10-14 04:34:36 +02:00
function encryptObjProperty(obj, model, map, key, self) {
var promises = [];
for (var prop in map) {
if (map.hasOwnProperty(prop)) {
/* jshint ignore:start */
2017-10-16 16:09:17 +02:00
(function (theProp, theModel) {
2017-10-14 04:34:36 +02:00
var promise = Q().then(function () {
2017-10-16 16:09:17 +02:00
var objProp = obj[(map[theProp] || theProp)];
if (objProp && objProp !== '') {
return self.cryptoService.encrypt(objProp, key);
2017-10-14 04:34:36 +02:00
}
return null;
}).then(function (val) {
2017-10-16 16:09:17 +02:00
theModel[theProp] = val;
2017-10-14 04:34:36 +02:00
});
promises.push(promise);
2017-10-16 16:09:17 +02:00
})(prop, model);
2017-10-14 04:34:36 +02:00
/* jshint ignore:end */
}
}
return Q.all(promises);
}
LoginService.prototype.get = function (id) {
var self = this,
key = null,
localData;
return self.userService.getUserIdPromise().then(function (userId) {
key = 'ciphers_' + userId;
return self.utilsService.getObjFromStorage(self.localDataKey);
}).then(function (data) {
localData = data;
if (!localData) {
localData = {};
}
return self.utilsService.getObjFromStorage(key);
}).then(function (ciphers) {
if (ciphers && id in ciphers) {
2017-10-16 16:09:17 +02:00
return new Cipher(ciphers[id], false, localData[id]);
}
return null;
2017-01-04 00:40:07 +01:00
});
};
LoginService.prototype.getAll = function () {
var self = this,
key = null,
localData = null;
2017-10-14 05:11:42 +02:00
return self.userService.getUserIdPromise().then(function (userId) {
key = 'ciphers_' + userId;
return self.utilsService.getObjFromStorage(self.localDataKey);
}).then(function (data) {
localData = data;
if (!localData) {
localData = {};
}
return self.utilsService.getObjFromStorage(key);
2017-10-14 05:11:42 +02:00
}).then(function (ciphers) {
var response = [];
2017-10-14 05:11:42 +02:00
for (var id in ciphers) {
if (id) {
2017-10-14 05:11:42 +02:00
response.push(new Cipher(ciphers[id], false, localData[id]));
2017-01-04 00:40:07 +01:00
}
}
2017-01-04 00:40:07 +01:00
return response;
2017-01-04 00:40:07 +01:00
});
};
LoginService.prototype.getAllDecrypted = function () {
2017-10-14 20:27:14 +02:00
if (this.decryptedCipherCache) {
return Q(this.decryptedCipherCache);
}
var deferred = Q.defer(),
decCiphers = [],
self = this;
2017-01-04 00:40:07 +01:00
2017-10-14 20:27:14 +02:00
self.cryptoService.getKey().then(function (key) {
2017-01-04 00:40:07 +01:00
if (!key) {
deferred.reject();
2017-10-14 20:27:14 +02:00
return true;
2017-01-04 00:40:07 +01:00
}
2017-10-14 20:27:14 +02:00
return self.getAll();
}).then(function (ciphers) {
if (ciphers === true) {
2017-01-04 00:40:07 +01:00
return;
}
var promises = [];
2017-10-14 05:11:42 +02:00
for (var i = 0; i < ciphers.length; i++) {
/* jshint ignore:start */
2017-10-14 05:11:42 +02:00
promises.push(ciphers[i].decrypt().then(function (cipher) {
decCiphers.push(cipher);
}));
/* jshint ignore:end */
}
2017-01-04 00:40:07 +01:00
return Q.all(promises);
2017-10-14 20:27:14 +02:00
}).then(function (stop) {
if (stop === true) {
return;
}
2017-10-14 05:11:42 +02:00
self.decryptedCipherCache = decCiphers;
2017-10-14 20:27:14 +02:00
deferred.resolve(self.decryptedCipherCache);
2017-01-04 00:40:07 +01:00
});
2017-10-14 20:27:14 +02:00
return deferred.promise;
2017-01-04 00:40:07 +01:00
};
LoginService.prototype.getAllDecryptedForFolder = function (folderId) {
return this.getAllDecrypted().then(function (ciphers) {
var ciphersToReturn = [];
for (var i = 0; i < ciphers.length; i++) {
if (ciphers[i].folderId === folderId) {
ciphersToReturn.push(ciphers[i]);
2017-01-04 00:40:07 +01:00
}
}
return ciphersToReturn;
2017-01-04 00:40:07 +01:00
});
};
LoginService.prototype.getAllDecryptedForDomain = function (domain) {
var self = this;
var eqDomainsPromise = self.settingsService.getEquivalentDomains().then(function (eqDomains) {
var matchingDomains = [];
for (var i = 0; i < eqDomains.length; i++) {
2017-01-18 03:43:26 +01:00
if (eqDomains[i].length && eqDomains[i].indexOf(domain) >= 0) {
matchingDomains = matchingDomains.concat(eqDomains[i]);
}
}
if (!matchingDomains.length) {
matchingDomains.push(domain);
}
return matchingDomains;
});
return Q.all([eqDomainsPromise, self.getAllDecrypted()]).then(function (result) {
var matchingDomains = result[0],
ciphers = result[1],
ciphersToReturn = [];
for (var i = 0; i < ciphers.length; i++) {
if (ciphers[i].domain && matchingDomains.indexOf(ciphers[i].domain) > -1) {
ciphersToReturn.push(ciphers[i]);
2017-01-04 00:40:07 +01:00
}
}
return ciphersToReturn;
2017-01-04 00:40:07 +01:00
});
};
2017-09-08 05:26:56 +02:00
LoginService.prototype.getLastUsedForDomain = function (domain) {
var self = this,
deferred = Q.defer();
self.getAllDecryptedForDomain(domain).then(function (ciphers) {
if (!ciphers.length) {
2017-09-08 05:26:56 +02:00
deferred.reject();
return;
}
var sortedCiphers = ciphers.sort(self.sortCiphersByLastUsed);
deferred.resolve(sortedCiphers[0]);
2017-09-08 05:26:56 +02:00
});
2017-09-08 05:26:56 +02:00
return deferred.promise;
};
LoginService.prototype.saveWithServer = function (cipher) {
2017-01-04 00:40:07 +01:00
var deferred = Q.defer();
var self = this,
2017-10-16 16:20:46 +02:00
request = new CipherRequest(cipher);
2017-01-04 00:40:07 +01:00
if (!cipher.id) {
self.apiService.postCipher(request).then(apiSuccess, function (response) {
deferred.reject(response);
2017-01-04 00:40:07 +01:00
});
}
else {
self.apiService.putCipher(cipher.id, request).then(apiSuccess, function (response) {
deferred.reject(response);
2017-01-04 00:40:07 +01:00
});
}
function apiSuccess(response) {
cipher.id = response.id;
self.userService.getUserIdPromise().then(function (userId) {
2017-10-16 16:09:17 +02:00
var data = new CipherData(response, userId);
return self.upsert(data);
}).then(function () {
deferred.resolve(cipher);
2017-01-04 00:40:07 +01:00
});
}
return deferred.promise;
};
LoginService.prototype.upsert = function (cipher) {
var self = this,
key = null;
return self.userService.getUserIdPromise().then(function (userId) {
key = 'ciphers_' + userId;
return self.utilsService.getObjFromStorage(key);
}).then(function (ciphers) {
if (!ciphers) {
ciphers = {};
}
2017-01-04 00:40:07 +01:00
if (cipher.constructor === Array) {
for (var i = 0; i < cipher.length; i++) {
ciphers[cipher[i].id] = cipher[i];
2017-01-04 00:40:07 +01:00
}
}
else {
ciphers[cipher.id] = cipher;
}
2017-01-04 00:40:07 +01:00
return self.utilsService.saveObjToStorage(key, ciphers);
}).then(function () {
self.decryptedCipherCache = null;
2017-01-04 00:40:07 +01:00
});
};
LoginService.prototype.updateLastUsedDate = function (id) {
var self = this;
var ciphersLocalData = null;
return self.utilsService.getObjFromStorage(self.localDataKey).then(function (obj) {
ciphersLocalData = obj;
if (!ciphersLocalData) {
ciphersLocalData = {};
}
if (ciphersLocalData[id]) {
ciphersLocalData[id].lastUsedDate = new Date().getTime();
}
else {
ciphersLocalData[id] = {
lastUsedDate: new Date().getTime()
};
}
return self.utilsService.saveObjToStorage(key, ciphersLocalData);
}).then(function () {
if (!self.decryptedCipherCache) {
return;
}
for (var i = 0; i < self.decryptedCipherCache.length; i++) {
if (self.decryptedCipherCache[i].id === id) {
self.decryptedCipherCache[i].localData = ciphersLocalData[id];
break;
}
}
});
};
LoginService.prototype.replace = function (ciphers) {
2017-01-04 00:40:07 +01:00
var self = this;
self.userService.getUserIdPromise().then(function (userId) {
return self.utilsService.saveObjToStorage('ciphers_' + userId, ciphers);
}).then(function () {
self.decryptedCipherCache = null;
2017-01-04 00:40:07 +01:00
});
};
LoginService.prototype.clear = function (userId) {
2017-01-04 00:40:07 +01:00
var self = this;
return self.utilsService.removeFromStorage('ciphers_' + userId).then(function () {
self.decryptedCipherCache = null;
2017-01-04 00:40:07 +01:00
});
};
LoginService.prototype.delete = function (id) {
var self = this,
key = null;
2017-01-04 00:40:07 +01:00
self.userService.getUserIdPromise().then(function () {
key = 'ciphers_' + userId;
return self.utilsService.getObjFromStorage(key);
}).then(function (logins) {
if (!logins) {
return null;
}
2017-01-04 00:40:07 +01:00
if (id.constructor === Array) {
for (var i = 0; i < id.length; i++) {
if (id[i] in logins) {
delete logins[id[i]];
2017-01-04 00:40:07 +01:00
}
}
}
else if (id in logins) {
delete logins[id];
}
else {
return null;
}
2017-01-04 00:40:07 +01:00
return logins;
}).then(function (logins) {
if (!logins) {
return false;
}
return self.utilsService.saveObjToStorage(key, logins);
}).then(function (clearCache) {
if (clearCache !== false) {
self.decryptedCipherCache = null;
}
2017-01-04 00:40:07 +01:00
});
};
LoginService.prototype.deleteWithServer = function (id) {
var self = this;
return self.apiService.deleteCipher(id).then(function () {
return self.delete(id);
2017-01-04 00:40:07 +01:00
});
};
LoginService.prototype.saveNeverDomain = function (domain) {
if (!domain) {
2017-10-14 03:39:31 +02:00
return Q();
}
2017-10-14 20:27:14 +02:00
var self = this;
return self.utilsService.getObjFromStorage(self.neverDomainsKey).then(function (domains) {
if (!domains) {
domains = {};
}
domains[domain] = null;
return self.utilsService.saveObjToStorage(key, domains);
});
};
LoginService.prototype.saveAttachmentWithServer = function (cipher, unencryptedFile) {
var deferred = Q.defer(),
self = this,
response = null,
data = null,
apiErrored = false;
2017-07-12 15:57:08 +02:00
var key, encFileName;
var reader = new FileReader();
reader.readAsArrayBuffer(unencryptedFile);
reader.onload = function (evt) {
self.cryptoService.getOrgKey(cipher.organizationId).then(function (theKey) {
2017-07-12 15:57:08 +02:00
key = theKey;
return self.cryptoService.encrypt(unencryptedFile.name, key);
}).then(function (fileName) {
encFileName = fileName;
return self.cryptoService.encryptToBytes(evt.target.result, key);
}).then(function (encData) {
var fd = new FormData();
var blob = new Blob([encData], { type: 'application/octet-stream' });
fd.append('data', blob, encFileName.encryptedString);
return self.apiService.postCipherAttachment(cipher.id, fd);
}).then(function (resp) {
response = resp;
return self.userService.getUserIdPromise();
}, function (resp) {
apiErrored = true;
handleErrorMessage(resp, deferred);
}).then(function (userId) {
if (apiErrored === true) {
return;
}
2017-10-16 16:09:17 +02:00
data = new CipherData(response, userId);
return self.upsert(data);
}).then(function () {
if (data) {
2017-10-16 16:09:17 +02:00
deferred.resolve(new CipherData(data));
}
2017-07-12 15:57:08 +02:00
});
};
reader.onerror = function (evt) {
deferred.reject('Error reading file.');
};
return deferred.promise;
};
LoginService.prototype.deleteAttachment = function (id, attachmentId) {
var self = this,
key = null;
self.userService.getUserIdPromise().then(function () {
key = 'ciphers_' + userId;
return self.utilsService.getObjFromStorage(key);
}).then(function (logins) {
if (logins && id in logins && logins[id].attachments) {
for (var i = 0; i < logins[id].attachments.length; i++) {
if (logins[id].attachments[i].id === attachmentId) {
logins[id].attachments.splice(i, 1);
2017-07-12 15:57:08 +02:00
}
}
return self.utilsService.saveObjToStorage(key, logins);
}
else {
return false;
}
}).then(function (clearCache) {
if (clearCache !== false) {
self.decryptedCipherCache = null;
}
2017-07-12 15:57:08 +02:00
});
};
LoginService.prototype.deleteAttachmentWithServer = function (id, attachmentId) {
var self = this,
deferred = Q.defer();
2017-07-12 15:57:08 +02:00
self.apiService.deleteCipherAttachment(id, attachmentId).then(function () {
return self.deleteAttachment(id, attachmentId);
2017-07-12 15:57:08 +02:00
}, function (response) {
2017-07-12 19:38:06 +02:00
handleErrorMessage(response, deferred);
return false;
}).then(function (apiSuccess) {
if (apiSuccess !== false) {
deferred.resolve();
}
2017-07-12 15:57:08 +02:00
});
return deferred.promise;
};
2017-08-30 14:43:06 +02:00
LoginService.prototype.sortLoginsByLastUsed = sortLoginsByLastUsed;
2017-08-30 14:43:06 +02:00
LoginService.prototype.sortLoginsByLastUsedThenName = function (a, b) {
var result = sortLoginsByLastUsed(a, b);
if (result !== 0) {
return result;
}
2017-08-30 14:43:06 +02:00
var nameA = (a.name + '_' + a.username).toUpperCase();
var nameB = (b.name + '_' + b.username).toUpperCase();
if (nameA < nameB) {
2017-08-29 19:03:08 +02:00
return -1;
}
2017-08-30 14:43:06 +02:00
if (nameA > nameB) {
2017-08-29 19:03:08 +02:00
return 1;
}
return 0;
};
2017-09-22 04:45:24 +02:00
function sortLoginsByLastUsed(a, b) {
2017-08-30 14:43:06 +02:00
var aLastUsed = a.localData && a.localData.lastUsedDate ? a.localData.lastUsedDate : null;
var bLastUsed = b.localData && b.localData.lastUsedDate ? b.localData.lastUsedDate : null;
2017-08-30 14:35:34 +02:00
2017-08-30 14:43:06 +02:00
if (aLastUsed && bLastUsed && aLastUsed < bLastUsed) {
return 1;
}
if (aLastUsed && !bLastUsed) {
return -1;
}
2017-08-30 14:35:34 +02:00
2017-08-30 14:43:06 +02:00
if (bLastUsed && aLastUsed && aLastUsed > bLastUsed) {
2017-08-30 14:35:34 +02:00
return -1;
}
2017-08-30 14:43:06 +02:00
if (bLastUsed && !aLastUsed) {
2017-08-30 14:35:34 +02:00
return 1;
}
return 0;
2017-08-30 14:43:06 +02:00
}
2017-08-30 14:35:34 +02:00
2017-01-04 00:40:07 +01:00
function handleError(error, deferred) {
deferred.reject(error);
}
2017-07-12 19:38:06 +02:00
function handleErrorMessage(error, deferred) {
if (error.validationErrors) {
for (var key in error.validationErrors) {
if (!error.validationErrors.hasOwnProperty(key)) {
continue;
}
if (error.validationErrors[key].length) {
deferred.reject(error.validationErrors[key][0]);
return;
}
}
}
deferred.reject(error.message);
return;
}
2017-07-14 21:34:05 +02:00
}