2017-10-13 23:07:20 +02:00
|
|
|
function LoginService(cryptoService, userService, apiService, settingsService, utilsService) {
|
2017-01-04 00:40:07 +01:00
|
|
|
this.cryptoService = cryptoService;
|
|
|
|
this.userService = userService;
|
|
|
|
this.apiService = apiService;
|
2017-01-14 19:43:59 +01:00
|
|
|
this.settingsService = settingsService;
|
2017-10-13 23:07:20 +02:00
|
|
|
this.utilsService = utilsService;
|
|
|
|
this.decryptedCipherCache = null;
|
|
|
|
this.localDataKey = 'sitesLocalData';
|
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 () {
|
2017-10-13 23:07:20 +02:00
|
|
|
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,
|
|
|
|
organizationId: login.organizationId
|
2017-01-04 00:40:07 +01:00
|
|
|
};
|
|
|
|
|
2017-04-24 19:58:32 +02:00
|
|
|
var orgKey = null;
|
2017-06-27 05:55:51 +02:00
|
|
|
return self.cryptoService.getOrgKey(login.organizationId).then(function (key) {
|
2017-04-24 19:58:32 +02:00
|
|
|
orgKey = key;
|
2017-06-27 05:55:51 +02:00
|
|
|
return self.cryptoService.encrypt(login.name, orgKey);
|
2017-04-24 19:58:32 +02:00
|
|
|
}).then(function (cs) {
|
2017-01-04 00:40:07 +01:00
|
|
|
model.name = cs;
|
2017-06-27 05:55:51 +02:00
|
|
|
return self.cryptoService.encrypt(login.uri, orgKey);
|
2017-01-04 00:40:07 +01:00
|
|
|
}).then(function (cs) {
|
|
|
|
model.uri = cs;
|
2017-06-27 05:55:51 +02:00
|
|
|
return self.cryptoService.encrypt(login.username, orgKey);
|
2017-01-04 00:40:07 +01:00
|
|
|
}).then(function (cs) {
|
|
|
|
model.username = cs;
|
2017-06-27 05:55:51 +02:00
|
|
|
return self.cryptoService.encrypt(login.password, orgKey);
|
2017-01-04 00:40:07 +01:00
|
|
|
}).then(function (cs) {
|
|
|
|
model.password = cs;
|
2017-06-27 05:55:51 +02:00
|
|
|
return self.cryptoService.encrypt(login.notes, orgKey);
|
2017-01-04 00:40:07 +01:00
|
|
|
}).then(function (cs) {
|
|
|
|
model.notes = cs;
|
2017-07-11 20:05:04 +02:00
|
|
|
return self.cryptoService.encrypt(login.totp, orgKey);
|
|
|
|
}).then(function (cs) {
|
|
|
|
model.totp = cs;
|
2017-09-22 04:45:24 +02:00
|
|
|
return self.encryptFields(login.fields, orgKey);
|
|
|
|
}).then(function (fields) {
|
|
|
|
model.fields = fields;
|
|
|
|
return model;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
LoginService.prototype.encryptFields = function (fields, key) {
|
|
|
|
var self = this;
|
|
|
|
if (!fields || !fields.length) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
};
|
|
|
|
|
|
|
|
return Q().then(function () {
|
|
|
|
if (!field.name || field.name === '') {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return self.cryptoService.encrypt(field.name, key);
|
|
|
|
}).then(function (cs) {
|
|
|
|
model.name = cs;
|
|
|
|
if (!field.value || field.value === '') {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return self.cryptoService.encrypt(field.value, key);
|
|
|
|
}).then(function (cs) {
|
|
|
|
model.value = cs;
|
2017-01-04 00:40:07 +01:00
|
|
|
return model;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
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) {
|
|
|
|
return new Login(ciphers[id], false, localData[id]);
|
|
|
|
}
|
2017-08-29 18:52:11 +02:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return null;
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
LoginService.prototype.getAll = function () {
|
|
|
|
var self = this,
|
|
|
|
key = null,
|
|
|
|
localData = null;
|
|
|
|
|
|
|
|
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 (logins) {
|
|
|
|
var response = [];
|
|
|
|
for (var id in logins) {
|
|
|
|
if (id) {
|
|
|
|
response.push(new Login(logins[id], false, localData[id]));
|
2017-01-04 00:40:07 +01:00
|
|
|
}
|
2017-10-13 23:07:20 +02:00
|
|
|
}
|
2017-01-04 00:40:07 +01:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return response;
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
LoginService.prototype.getAllDecrypted = function () {
|
2017-10-13 23:07:20 +02:00
|
|
|
var self = this,
|
|
|
|
decLogins = [];
|
2017-01-04 00:40:07 +01:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return self.cryptoService.getKey().then(function (key) {
|
2017-01-04 00:40:07 +01:00
|
|
|
if (!key) {
|
|
|
|
deferred.reject();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
if (self.decryptedCipherCache) {
|
|
|
|
deferred.resolve(self.decryptedCipherCache);
|
2017-01-04 00:40:07 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return self.getAll();
|
|
|
|
}).then(function (logins) {
|
2017-01-04 00:40:07 +01:00
|
|
|
var promises = [];
|
2017-10-13 23:07:20 +02:00
|
|
|
for (var i = 0; i < logins.length; i++) {
|
|
|
|
/* jshint ignore:start */
|
|
|
|
promises.push(logins[i].decrypt().then(function (login) {
|
|
|
|
decLogins.push(login);
|
|
|
|
}));
|
|
|
|
/* jshint ignore:end */
|
|
|
|
}
|
2017-01-04 00:40:07 +01:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return Q.all(promises);
|
|
|
|
}).then(function () {
|
|
|
|
self.decryptedCipherCache = decLogins;
|
|
|
|
return self.decryptedCipherCache;
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
LoginService.prototype.getAllDecryptedForFolder = function (folderId) {
|
2017-10-13 23:07:20 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return ciphersToReturn;
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
LoginService.prototype.getAllDecryptedForDomain = function (domain) {
|
|
|
|
var self = this;
|
|
|
|
|
2017-01-14 19:43:59 +01:00
|
|
|
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]);
|
2017-01-14 19:43:59 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-14 23:58:22 +01:00
|
|
|
if (!matchingDomains.length) {
|
|
|
|
matchingDomains.push(domain);
|
|
|
|
}
|
|
|
|
|
2017-01-14 19:43:59 +01:00
|
|
|
return matchingDomains;
|
|
|
|
});
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return Q.all([eqDomainsPromise, self.getAllDecrypted()]).then(function (result) {
|
|
|
|
var matchingDomains = result[0],
|
|
|
|
ciphers = result[1],
|
|
|
|
ciphersToReturn = [];
|
2017-01-14 19:43:59 +01:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return ciphersToReturn;
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-09-08 05:26:56 +02:00
|
|
|
LoginService.prototype.getLastUsedForDomain = function (domain) {
|
2017-10-13 23:07:20 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
var sortedCiphers = ciphers.sort(self.sortCiphersByLastUsed);
|
|
|
|
deferred.resolve(sortedCiphers[0]);
|
2017-09-08 05:26:56 +02:00
|
|
|
});
|
2017-10-13 23:07:20 +02:00
|
|
|
|
2017-09-08 05:26:56 +02:00
|
|
|
return deferred.promise;
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
LoginService.prototype.saveWithServer = function (cipher) {
|
2017-01-04 00:40:07 +01:00
|
|
|
var deferred = Q.defer();
|
|
|
|
|
|
|
|
var self = this,
|
2017-10-13 23:07:20 +02:00
|
|
|
// TODO
|
|
|
|
request = new CipherRequest(cipher, 1); // 1 = Login
|
2017-01-04 00:40:07 +01:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
if (!cipher.id) {
|
|
|
|
self.apiService.postCipher(request).then(apiSuccess, function (response) {
|
|
|
|
deferred.reject(response);
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
else {
|
2017-10-13 23:07:20 +02:00
|
|
|
self.apiService.putCipher(cipher.id, request).then(apiSuccess, function (response) {
|
|
|
|
deferred.reject(response);
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function apiSuccess(response) {
|
2017-10-13 23:07:20 +02:00
|
|
|
cipher.id = response.id;
|
|
|
|
self.userService.getUserIdPromise().then(function (userId) {
|
2017-01-04 00:40:07 +01:00
|
|
|
var data = new LoginData(response, userId);
|
2017-10-13 23:07:20 +02:00
|
|
|
return self.upsert(data);
|
|
|
|
}).then(function () {
|
|
|
|
deferred.resolve(cipher);
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return deferred.promise;
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
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
|
|
|
|
2017-10-13 23:07:20 +02: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
|
|
|
}
|
2017-10-13 23:07:20 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
ciphers[cipher.id] = cipher;
|
|
|
|
}
|
2017-01-04 00:40:07 +01:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return self.utilsService.saveObjToStorage(key, ciphers);
|
|
|
|
}).then(function () {
|
|
|
|
self.decryptedCipherCache = null;
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
LoginService.prototype.updateLastUsedDate = function (id) {
|
2017-08-29 18:52:11 +02:00
|
|
|
var self = this;
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
var ciphersLocalData = null;
|
|
|
|
return self.utilsService.getObjFromStorage(self.localDataKey).then(function (obj) {
|
|
|
|
ciphersLocalData = obj;
|
|
|
|
|
|
|
|
if (!ciphersLocalData) {
|
|
|
|
ciphersLocalData = {};
|
2017-08-29 18:52:11 +02:00
|
|
|
}
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
if (ciphersLocalData[id]) {
|
|
|
|
ciphersLocalData[id].lastUsedDate = new Date().getTime();
|
2017-08-29 18:52:11 +02:00
|
|
|
}
|
|
|
|
else {
|
2017-10-13 23:07:20 +02:00
|
|
|
ciphersLocalData[id] = {
|
2017-08-29 18:52:11 +02:00
|
|
|
lastUsedDate: new Date().getTime()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
return self.utilsService.saveObjToStorage(key, ciphersLocalData);
|
|
|
|
}).then(function () {
|
|
|
|
if (!self.decryptedCipherCache) {
|
|
|
|
return;
|
|
|
|
}
|
2017-08-29 18:52:11 +02:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
for (var i = 0; i < self.decryptedCipherCache.length; i++) {
|
|
|
|
if (self.decryptedCipherCache[i].id === id) {
|
|
|
|
self.decryptedCipherCache[i].localData = ciphersLocalData[id];
|
|
|
|
break;
|
2017-08-29 18:52:11 +02:00
|
|
|
}
|
2017-10-13 23:07:20 +02:00
|
|
|
}
|
2017-08-29 18:52:11 +02:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
LoginService.prototype.replace = function (ciphers) {
|
2017-01-04 00:40:07 +01:00
|
|
|
var self = this;
|
2017-10-13 23:07:20 +02:00
|
|
|
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
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
LoginService.prototype.clear = function (userId) {
|
2017-01-04 00:40:07 +01:00
|
|
|
var self = this;
|
2017-10-13 23:07:20 +02:00
|
|
|
return self.utilsService.removeFromStorage('ciphers_' + userId).then(function () {
|
|
|
|
self.decryptedCipherCache = null;
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
LoginService.prototype.delete = function (id) {
|
|
|
|
var self = this,
|
|
|
|
key = null;
|
2017-01-04 00:40:07 +01:00
|
|
|
|
2017-10-13 23:07:20 +02: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
|
|
|
|
2017-10-13 23:07:20 +02: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
|
|
|
}
|
|
|
|
}
|
2017-10-13 23:07:20 +02:00
|
|
|
}
|
|
|
|
else if (id in logins) {
|
|
|
|
delete logins[id];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return null;
|
|
|
|
}
|
2017-01-04 00:40:07 +01:00
|
|
|
|
2017-10-13 23:07:20 +02: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;
|
2017-10-13 23:07:20 +02:00
|
|
|
return self.apiService.deleteCipher(id).then(function () {
|
|
|
|
return self.delete(id);
|
2017-01-04 00:40:07 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-06-13 19:59:16 +02:00
|
|
|
LoginService.prototype.saveNeverDomain = function (domain) {
|
|
|
|
if (!domain) {
|
2017-10-14 03:39:31 +02:00
|
|
|
return Q();
|
2017-06-13 19:59:16 +02:00
|
|
|
}
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
var key = 'neverDomains';
|
|
|
|
return self.utilsService.getObjFromStorage(key).then(function (domains) {
|
|
|
|
if (!domains) {
|
|
|
|
domains = {};
|
|
|
|
}
|
2017-06-13 19:59:16 +02:00
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
domains[domain] = null;
|
|
|
|
return self.utilsService.saveObjToStorage(key, domains);
|
|
|
|
});
|
2017-06-13 19:59:16 +02:00
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
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) {
|
2017-10-13 23:07:20 +02:00
|
|
|
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);
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = new LoginData(response, userId);
|
|
|
|
return self.upsert(data);
|
|
|
|
}).then(function () {
|
|
|
|
if (data) {
|
|
|
|
deferred.resolve(new Login(data));
|
|
|
|
}
|
2017-07-12 15:57:08 +02:00
|
|
|
});
|
|
|
|
};
|
|
|
|
reader.onerror = function (evt) {
|
|
|
|
deferred.reject('Error reading file.');
|
|
|
|
};
|
|
|
|
|
|
|
|
return deferred.promise;
|
|
|
|
};
|
|
|
|
|
2017-10-13 23:07:20 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
2017-10-13 23:07:20 +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) {
|
2017-10-13 23:07:20 +02:00
|
|
|
var self = this,
|
|
|
|
deferred = Q.defer();
|
2017-07-12 15:57:08 +02:00
|
|
|
|
2017-10-13 23:07:20 +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);
|
2017-10-13 23:07:20 +02:00
|
|
|
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-29 18:52:11 +02:00
|
|
|
|
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-29 18:52:11 +02:00
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2017-08-29 18:52:11 +02:00
|
|
|
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
|
|
|
}
|