2019-07-30 07:35:42 +00:00
|
|
|
/* eslint-disable no-unused-vars */
|
|
|
|
|
2017-12-13 18:57:40 +00:00
|
|
|
require('app-module-path').addPath(__dirname);
|
|
|
|
|
|
|
|
const { time } = require('lib/time-utils.js');
|
2019-09-23 22:23:10 +00:00
|
|
|
const { asyncTest, fileContentEqual, setupDatabase, setupDatabaseAndSynchronizer, db, synchronizer, fileApi, sleep, clearDatabase, switchClient, syncTargetId, objectsEqual, checkThrowAsync } = require('test-utils.js');
|
2017-12-14 18:12:14 +00:00
|
|
|
const Folder = require('lib/models/Folder.js');
|
|
|
|
const Note = require('lib/models/Note.js');
|
|
|
|
const Tag = require('lib/models/Tag.js');
|
2017-12-13 18:57:40 +00:00
|
|
|
const { Database } = require('lib/database.js');
|
2017-12-14 18:12:14 +00:00
|
|
|
const Setting = require('lib/models/Setting.js');
|
|
|
|
const BaseItem = require('lib/models/BaseItem.js');
|
|
|
|
const BaseModel = require('lib/BaseModel.js');
|
2017-12-13 18:57:40 +00:00
|
|
|
const MasterKey = require('lib/models/MasterKey');
|
|
|
|
const SyncTargetRegistry = require('lib/SyncTargetRegistry.js');
|
|
|
|
const EncryptionService = require('lib/services/EncryptionService.js');
|
|
|
|
|
|
|
|
process.on('unhandledRejection', (reason, p) => {
|
|
|
|
console.log('Unhandled Rejection at: Promise', p, 'reason:', reason);
|
|
|
|
});
|
|
|
|
|
|
|
|
jasmine.DEFAULT_TIMEOUT_INTERVAL = 15000; // The first test is slow because the database needs to be built
|
|
|
|
|
|
|
|
let service = null;
|
|
|
|
|
|
|
|
describe('Encryption', function() {
|
|
|
|
|
|
|
|
beforeEach(async (done) => {
|
|
|
|
await setupDatabaseAndSynchronizer(1);
|
2019-05-06 20:35:29 +00:00
|
|
|
await switchClient(1);
|
2017-12-13 18:57:40 +00:00
|
|
|
service = new EncryptionService();
|
|
|
|
BaseItem.encryptionService_ = service;
|
|
|
|
Setting.setValue('encryption.enabled', true);
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2019-09-23 22:23:10 +00:00
|
|
|
it('should encode and decode header', asyncTest(async () => {
|
2017-12-13 18:57:40 +00:00
|
|
|
const header = {
|
|
|
|
encryptionMethod: EncryptionService.METHOD_SJCL,
|
|
|
|
masterKeyId: '01234568abcdefgh01234568abcdefgh',
|
|
|
|
};
|
|
|
|
|
|
|
|
const encodedHeader = service.encodeHeader_(header);
|
2020-01-22 22:01:58 +00:00
|
|
|
const decodedHeader = service.decodeHeaderBytes_(encodedHeader);
|
2017-12-13 18:57:40 +00:00
|
|
|
delete decodedHeader.length;
|
|
|
|
|
|
|
|
expect(objectsEqual(header, decodedHeader)).toBe(true);
|
2019-09-23 22:23:10 +00:00
|
|
|
}));
|
2017-12-13 18:57:40 +00:00
|
|
|
|
2019-09-23 22:23:10 +00:00
|
|
|
it('should generate and decrypt a master key', asyncTest(async () => {
|
2017-12-13 18:57:40 +00:00
|
|
|
const masterKey = await service.generateMasterKey('123456');
|
|
|
|
expect(!!masterKey.checksum).toBe(true);
|
|
|
|
expect(!!masterKey.content).toBe(true);
|
|
|
|
|
|
|
|
let hasThrown = false;
|
|
|
|
try {
|
2020-01-22 22:01:58 +00:00
|
|
|
await service.decryptMasterKey_(masterKey, 'wrongpassword');
|
2017-12-13 18:57:40 +00:00
|
|
|
} catch (error) {
|
|
|
|
hasThrown = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
expect(hasThrown).toBe(true);
|
|
|
|
|
2020-01-22 22:01:58 +00:00
|
|
|
const decryptedMasterKey = await service.decryptMasterKey_(masterKey, '123456');
|
2017-12-13 18:57:40 +00:00
|
|
|
expect(decryptedMasterKey.length).toBe(512);
|
2019-09-23 22:23:10 +00:00
|
|
|
}));
|
2017-12-13 18:57:40 +00:00
|
|
|
|
2019-09-23 22:23:10 +00:00
|
|
|
it('should encrypt and decrypt with a master key', asyncTest(async () => {
|
2017-12-13 18:57:40 +00:00
|
|
|
let masterKey = await service.generateMasterKey('123456');
|
|
|
|
masterKey = await MasterKey.save(masterKey);
|
|
|
|
|
2020-01-22 22:01:58 +00:00
|
|
|
await service.loadMasterKey_(masterKey, '123456', true);
|
2017-12-13 18:57:40 +00:00
|
|
|
|
|
|
|
const cipherText = await service.encryptString('some secret');
|
|
|
|
const plainText = await service.decryptString(cipherText);
|
|
|
|
|
|
|
|
expect(plainText).toBe('some secret');
|
|
|
|
|
|
|
|
// Test that a long string, that is going to be split into multiple chunks, encrypt
|
|
|
|
// and decrypt properly too.
|
|
|
|
let veryLongSecret = '';
|
|
|
|
for (let i = 0; i < service.chunkSize() * 3; i++) veryLongSecret += Math.floor(Math.random() * 9);
|
|
|
|
|
|
|
|
const cipherText2 = await service.encryptString(veryLongSecret);
|
|
|
|
const plainText2 = await service.decryptString(cipherText2);
|
|
|
|
|
|
|
|
expect(plainText2 === veryLongSecret).toBe(true);
|
2019-09-23 22:23:10 +00:00
|
|
|
}));
|
2017-12-13 18:57:40 +00:00
|
|
|
|
2020-01-22 22:01:58 +00:00
|
|
|
it('should decrypt various encryption methods', asyncTest(async () => {
|
|
|
|
let masterKey = await service.generateMasterKey('123456');
|
|
|
|
masterKey = await MasterKey.save(masterKey);
|
|
|
|
await service.loadMasterKey_(masterKey, '123456', true);
|
|
|
|
|
|
|
|
{
|
|
|
|
const cipherText = await service.encryptString('some secret', {
|
|
|
|
encryptionMethod: EncryptionService.METHOD_SJCL_2,
|
|
|
|
});
|
|
|
|
const plainText = await service.decryptString(cipherText);
|
|
|
|
expect(plainText).toBe('some secret');
|
|
|
|
const header = await service.decodeHeaderString(cipherText);
|
|
|
|
expect(header.encryptionMethod).toBe(EncryptionService.METHOD_SJCL_2);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
const cipherText = await service.encryptString('some secret', {
|
|
|
|
encryptionMethod: EncryptionService.METHOD_SJCL_3,
|
|
|
|
});
|
|
|
|
const plainText = await service.decryptString(cipherText);
|
|
|
|
expect(plainText).toBe('some secret');
|
|
|
|
const header = await service.decodeHeaderString(cipherText);
|
|
|
|
expect(header.encryptionMethod).toBe(EncryptionService.METHOD_SJCL_3);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
2019-09-23 22:23:10 +00:00
|
|
|
it('should fail to decrypt if master key not present', asyncTest(async () => {
|
2017-12-13 18:57:40 +00:00
|
|
|
let masterKey = await service.generateMasterKey('123456');
|
|
|
|
masterKey = await MasterKey.save(masterKey);
|
|
|
|
|
2020-01-22 22:01:58 +00:00
|
|
|
await service.loadMasterKey_(masterKey, '123456', true);
|
2017-12-13 18:57:40 +00:00
|
|
|
|
|
|
|
const cipherText = await service.encryptString('some secret');
|
|
|
|
|
|
|
|
await service.unloadMasterKey(masterKey);
|
|
|
|
|
|
|
|
let hasThrown = await checkThrowAsync(async () => await service.decryptString(cipherText));
|
|
|
|
|
|
|
|
expect(hasThrown).toBe(true);
|
2019-09-23 22:23:10 +00:00
|
|
|
}));
|
2017-12-13 18:57:40 +00:00
|
|
|
|
|
|
|
|
2019-09-23 22:23:10 +00:00
|
|
|
it('should fail to decrypt if data tampered with', asyncTest(async () => {
|
2017-12-13 18:57:40 +00:00
|
|
|
let masterKey = await service.generateMasterKey('123456');
|
|
|
|
masterKey = await MasterKey.save(masterKey);
|
|
|
|
|
2020-01-22 22:01:58 +00:00
|
|
|
await service.loadMasterKey_(masterKey, '123456', true);
|
2017-12-13 18:57:40 +00:00
|
|
|
|
|
|
|
let cipherText = await service.encryptString('some secret');
|
2019-07-30 07:35:42 +00:00
|
|
|
cipherText += 'ABCDEFGHIJ';
|
2017-12-13 18:57:40 +00:00
|
|
|
|
|
|
|
let hasThrown = await checkThrowAsync(async () => await service.decryptString(cipherText));
|
|
|
|
|
|
|
|
expect(hasThrown).toBe(true);
|
2019-09-23 22:23:10 +00:00
|
|
|
}));
|
2017-12-13 18:57:40 +00:00
|
|
|
|
2019-09-23 22:23:10 +00:00
|
|
|
it('should encrypt and decrypt notes and folders', asyncTest(async () => {
|
2017-12-13 18:57:40 +00:00
|
|
|
let masterKey = await service.generateMasterKey('123456');
|
|
|
|
masterKey = await MasterKey.save(masterKey);
|
2020-01-22 22:01:58 +00:00
|
|
|
await service.loadMasterKey_(masterKey, '123456', true);
|
2017-12-13 18:57:40 +00:00
|
|
|
|
|
|
|
let folder = await Folder.save({ title: 'folder' });
|
|
|
|
let note = await Note.save({ title: 'encrypted note', body: 'something', parent_id: folder.id });
|
|
|
|
let serialized = await Note.serializeForSync(note);
|
|
|
|
let deserialized = Note.filter(await Note.unserialize(serialized));
|
|
|
|
|
|
|
|
// Check that required properties are not encrypted
|
|
|
|
expect(deserialized.id).toBe(note.id);
|
|
|
|
expect(deserialized.parent_id).toBe(note.parent_id);
|
|
|
|
expect(deserialized.updated_time).toBe(note.updated_time);
|
|
|
|
|
|
|
|
// Check that at least title and body are encrypted
|
|
|
|
expect(!deserialized.title).toBe(true);
|
|
|
|
expect(!deserialized.body).toBe(true);
|
|
|
|
|
|
|
|
// Check that encrypted data is there
|
|
|
|
expect(!!deserialized.encryption_cipher_text).toBe(true);
|
|
|
|
|
2019-07-30 07:35:42 +00:00
|
|
|
const encryptedNote = await Note.save(deserialized);
|
|
|
|
const decryptedNote = await Note.decrypt(encryptedNote);
|
2017-12-13 18:57:40 +00:00
|
|
|
|
|
|
|
expect(decryptedNote.title).toBe(note.title);
|
|
|
|
expect(decryptedNote.body).toBe(note.body);
|
|
|
|
expect(decryptedNote.id).toBe(note.id);
|
|
|
|
expect(decryptedNote.parent_id).toBe(note.parent_id);
|
2019-09-23 22:23:10 +00:00
|
|
|
}));
|
2017-12-13 18:57:40 +00:00
|
|
|
|
2019-09-23 22:23:10 +00:00
|
|
|
it('should encrypt and decrypt files', asyncTest(async () => {
|
2017-12-18 19:54:03 +00:00
|
|
|
let masterKey = await service.generateMasterKey('123456');
|
|
|
|
masterKey = await MasterKey.save(masterKey);
|
2020-01-22 22:01:58 +00:00
|
|
|
await service.loadMasterKey_(masterKey, '123456', true);
|
2017-12-18 19:54:03 +00:00
|
|
|
|
2019-09-19 21:51:18 +00:00
|
|
|
const sourcePath = `${__dirname}/../tests/support/photo.jpg`;
|
|
|
|
const encryptedPath = `${__dirname}/data/photo.crypted`;
|
|
|
|
const decryptedPath = `${__dirname}/data/photo.jpg`;
|
2017-12-18 19:54:03 +00:00
|
|
|
|
|
|
|
await service.encryptFile(sourcePath, encryptedPath);
|
|
|
|
await service.decryptFile(encryptedPath, decryptedPath);
|
|
|
|
|
|
|
|
expect(fileContentEqual(sourcePath, encryptedPath)).toBe(false);
|
|
|
|
expect(fileContentEqual(sourcePath, decryptedPath)).toBe(true);
|
2019-09-23 22:23:10 +00:00
|
|
|
}));
|
2017-12-18 19:54:03 +00:00
|
|
|
|
2020-01-22 22:01:58 +00:00
|
|
|
// it('should upgrade master key encryption mode', asyncTest(async () => {
|
|
|
|
// let masterKey = await service.generateMasterKey('123456', {
|
|
|
|
// encryptionMethod: EncryptionService.METHOD_SJCL_2,
|
|
|
|
// });
|
|
|
|
// masterKey = await MasterKey.save(masterKey);
|
|
|
|
// Setting.setObjectKey('encryption.passwordCache', masterKey.id, '123456');
|
|
|
|
// Setting.setValue('encryption.activeMasterKeyId', masterKey.id);
|
|
|
|
|
|
|
|
// await sleep(0.01);
|
|
|
|
|
|
|
|
// await service.loadMasterKeysFromSettings();
|
|
|
|
|
|
|
|
// masterKeyNew = await MasterKey.load(masterKey.id);
|
|
|
|
|
|
|
|
// // Check that the master key has been upgraded
|
|
|
|
|
|
|
|
// expect(masterKeyNew.created_time).toBe(masterKey.created_time);
|
|
|
|
// expect(masterKeyNew.updated_time === masterKey.updated_time).toBe(false);
|
|
|
|
// expect(masterKeyNew.content === masterKey.content).toBe(false);
|
|
|
|
// expect(masterKeyNew.encryption_method === masterKey.encryption_method).toBe(false);
|
|
|
|
// expect(masterKeyNew.checksum === masterKey.checksum).toBe(false);
|
|
|
|
// expect(masterKeyNew.encryption_method).toBe(service.defaultMasterKeyEncryptionMethod_);
|
|
|
|
|
|
|
|
// // Check that encryption still works
|
|
|
|
|
|
|
|
// const cipherText = await service.encryptString('some secret');
|
|
|
|
// const plainText = await service.decryptString(cipherText);
|
|
|
|
// expect(plainText).toBe('some secret');
|
|
|
|
// }));
|
|
|
|
|
2019-07-30 07:35:42 +00:00
|
|
|
});
|