const childProcess = require('child_process'); const path = require('path'); const fs = require('fs'); const { isWindowsOS } = require('../js/utils/misc.js'); let executionPath = null; let userConfigDir = null; let searchConfig; let SearchApi; jest.mock('electron', function() { return { app: { getPath: mockedGetPath } } }); function mockedGetPath(type) { if (type === 'exe') { return executionPath; } if (type === 'userData') { return userConfigDir } return ''; } describe('Tests for Search', function() { let userId; let key; let dataFolderPath; let realTimeIndexPath; let tempBatchPath; let currentDate = new Date().getTime(); jasmine.DEFAULT_TIMEOUT_INTERVAL = 90000; beforeAll(function (done) { childProcess.exec(`npm rebuild --target=${process.version} --build-from-source`, function(err) { userId = 12345678910112; key = 'jjjehdnctsjyieoalskcjdhsnahsadndfnusdfsdfsd='; executionPath = path.join(__dirname, 'library'); if (isWindowsOS) { executionPath = path.join(__dirname, '..', 'library'); } userConfigDir = path.join(__dirname, '..'); searchConfig = require('../js/search/searchConfig.js'); const { Search } = require('../js/search/search.js'); SearchApi = new Search(userId, key); realTimeIndexPath = path.join(userConfigDir, 'data', 'temp_realtime_index'); tempBatchPath = path.join(userConfigDir, 'data', 'temp_batch_indexes'); dataFolderPath = path.join(userConfigDir, 'data'); if (fs.existsSync(dataFolderPath)) { deleteIndexFolders(dataFolderPath) } done(); }); }); afterAll(function (done) { setTimeout(function () { deleteIndexFolders(dataFolderPath); let root = path.join(userConfigDir, `${searchConfig.FOLDERS_CONSTANTS.PREFIX_NAME}_${userId}.enc`); if (fs.existsSync(root)) { fs.unlinkSync(root); } done(); }, 3000); }); function deleteIndexFolders(location) { if (fs.existsSync(location)) { fs.readdirSync(location).forEach(function(file) { let curPath = path.join(location, file); if (fs.lstatSync(curPath).isDirectory()) { deleteIndexFolders(curPath); } else { fs.unlinkSync(curPath); } }); fs.rmdirSync(location); } } describe('Search Initial checks', function() { it('should be initialized', function (done) { setTimeout(function () { expect(SearchApi.isInitialized).toBe(true); expect(SearchApi.indexFolderName).toBe(`${searchConfig.FOLDERS_CONSTANTS.PREFIX_NAME_PATH}_${userId}`); expect(SearchApi.dataFolder).toBe(searchConfig.FOLDERS_CONSTANTS.INDEX_PATH); expect(SearchApi.realTimeIndex).toBe(searchConfig.FOLDERS_CONSTANTS.TEMP_REAL_TIME_INDEX); expect(SearchApi.batchIndex).toBe(searchConfig.FOLDERS_CONSTANTS.TEMP_BATCH_INDEX_FOLDER); expect(SearchApi.messageData).toEqual([]); expect(SearchApi.isRealTimeIndexing).toBe(false); done(); }, 3000) }); it('should isLibInit to true', function () { let init = SearchApi.isLibInit(); expect(init).toEqual(true); }); it('should isLibInit to false', function () { SearchApi.isInitialized = false; let init = SearchApi.isLibInit(); expect(init).toEqual(false); SearchApi.isInitialized = true; }); it('should exist index folder', function() { expect(fs.existsSync(path.join(userConfigDir, 'data', 'search_index_12345678910112'))).toBe(true); expect(fs.existsSync(realTimeIndexPath)).toBe(true); }); it('should not exist index folder', function() { expect(fs.existsSync(tempBatchPath)).toBe(false); }); }); describe('Batch indexing process tests', function () { it('should index in a batch', function (done) { let messages = [{ messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "it works" }, { messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "it works" }, { messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "it works" }]; const indexBatch = jest.spyOn(SearchApi, 'indexBatch'); SearchApi.indexBatch(JSON.stringify(messages)).then(function () { expect(fs.existsSync(tempBatchPath)).toBe(true); expect(indexBatch).toHaveBeenCalledWith(JSON.stringify(messages)); done(); }); }); it('should not batch index', function (done) { const indexBatch = jest.spyOn(SearchApi, 'indexBatch'); SearchApi.indexBatch().catch(function (err) { expect(indexBatch).toHaveBeenCalled(); expect(err).toBeTruthy(); done(); }); }); it('should not batch index invalid object', function (done) { const indexBatch = jest.spyOn(SearchApi, 'indexBatch'); SearchApi.indexBatch('message').catch(function (err) { expect(err).toBeTruthy(); expect(indexBatch).toHaveBeenCalledWith('message'); done(); }); }); it('should not batch index parse error', function (done) { let message = { messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "it works" }; const indexBatch = jest.spyOn(SearchApi, 'indexBatch'); SearchApi.indexBatch(JSON.stringify(message)).catch(function (err) { expect(err).toBeTruthy(); expect(indexBatch).toHaveBeenCalled(); done(); }); }); it('should not batch index isInitialized is false', function (done) { SearchApi.isInitialized = false; let message = [ { messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "it fails" } ]; const indexBatch = jest.spyOn(SearchApi, 'indexBatch'); SearchApi.indexBatch(JSON.stringify(message)).catch(function (err) { expect(err).toBeTruthy(); expect(indexBatch).toHaveBeenCalledWith(JSON.stringify(message)); SearchApi.isInitialized = true; done(); }); }); it('should match messages length after batch indexing', function (done) { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); SearchApi.searchQuery('it works', [], [], '', undefined, undefined, 25, 0, 0).then(function (res) { expect(res.messages.length).toEqual(0); expect(searchQuery).toHaveBeenCalled(); done() }); }); it('should merge batch index to user index', function (done) { const mergeIndexBatches = jest.spyOn(SearchApi, 'mergeIndexBatches'); SearchApi.mergeIndexBatches().then(function () { expect(fs.existsSync(tempBatchPath)).toBe(false); expect(mergeIndexBatches).toHaveBeenCalled(); done(); }); }); it('should match messages length after batch indexing', function (done) { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); SearchApi.searchQuery('it works', [], [], '', undefined, undefined, 25, 0, 0).then(function (res) { expect(res.messages.length).toEqual(3); expect(searchQuery).toHaveBeenCalled(); done(); }); }); }); describe('RealTime indexing process', function () { it('should index realTime message', function () { let message = [{ messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "realtime working" }]; const batchRealTimeIndexing = jest.spyOn(SearchApi, 'batchRealTimeIndexing'); SearchApi.batchRealTimeIndexing(message); expect(batchRealTimeIndexing).toHaveBeenCalled(); }); it('should match message length', function (done) { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); SearchApi.searchQuery('realtime working', ["71811853189212"], ["Au8O2xKHyX1LtE6zW019GX///rZYegAtdA=="], '', undefined, undefined, 25, 0, 0).then(function (res) { expect(res.messages.length).toEqual(3); expect(fs.existsSync(realTimeIndexPath)).toBe(true); expect(searchQuery).toHaveBeenCalled(); done(); }) }); it('should not index realTime message', function (done) { let message = [{ messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "isRealTimeIndexing" }]; const batchRealTimeIndexing = jest.spyOn(SearchApi, 'batchRealTimeIndexing'); const realTimeIndexing = jest.spyOn(SearchApi, 'realTimeIndexing'); SearchApi.isRealTimeIndexing = true; expect(SearchApi.checkIsRealTimeIndexing()).toBe(true); SearchApi.batchRealTimeIndexing(message); expect(batchRealTimeIndexing).toHaveBeenCalled(); expect(realTimeIndexing).not.toBeCalled(); setTimeout(function () { SearchApi.searchQuery('isRealTimeIndexing', [], [], '', undefined, undefined, 25, 0, 0).then(function (res) { expect(res.messages.length).toEqual(0); expect(fs.existsSync(realTimeIndexPath)).toBe(true); done(); }); }, 6000) }); it('should not call the real-time index', function () { let message = [{ messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "isRealTimeIndexing" }]; const batchRealTimeIndexing = jest.spyOn(SearchApi, 'batchRealTimeIndexing'); const realTimeIndexing = jest.spyOn(SearchApi, 'realTimeIndexing'); SearchApi.isRealTimeIndexing = true; SearchApi.batchRealTimeIndexing(message); expect(batchRealTimeIndexing).toHaveBeenCalled(); expect(realTimeIndexing).not.toBeCalled(); }); it('should not realTime index invalid object', function () { let message = [{ messageId: "Jc+4K8RtPxHJfyuDQU9atX///qN3KHYXdA==", threadId: "Au8O2xKHyX1LtE6zW019GX///rZYegAtdA==", ingestionDate: currentDate.toString(), senderId: "71811853189212", chatType: "CHATROOM", isPublic: "false", sendingApp: "lc", text: "isRealTimeIndexing" }]; const realTimeIndexing = jest.spyOn(SearchApi, 'realTimeIndexing'); expect(function () { SearchApi.realTimeIndexing('message') }).toThrow(); expect(function () { SearchApi.realTimeIndexing() }).toThrow(); SearchApi.isInitialized = false; expect(function () { SearchApi.realTimeIndexing(JSON.stringify(message)) }).toThrow(new Error('Library not initialized')); SearchApi.isInitialized = true; expect(realTimeIndexing).toHaveBeenCalled(); expect(realTimeIndexing).toHaveBeenCalledTimes(3); }); it('should return realTime bool', function () { const checkIsRealTimeIndexing = jest.spyOn(SearchApi, 'checkIsRealTimeIndexing'); SearchApi.isRealTimeIndexing = true; expect(SearchApi.checkIsRealTimeIndexing()).toBe(true); SearchApi.isRealTimeIndexing = false; expect(SearchApi.checkIsRealTimeIndexing()).toBe(false); expect(checkIsRealTimeIndexing).toHaveBeenCalled(); expect(checkIsRealTimeIndexing).toHaveBeenCalledTimes(2); }); it('should delete realtime index', function () { const deleteRealTimeFolder = jest.spyOn(SearchApi, 'deleteRealTimeFolder'); SearchApi.deleteRealTimeFolder(); expect(fs.existsSync(realTimeIndexPath)).toBe(true); expect(deleteRealTimeFolder).toHaveBeenCalled(); }); }); describe('Test for encryption of the index', function () { it('should encrypt user index', function (done) { const encryptIndex = jest.spyOn(SearchApi, 'encryptIndex'); SearchApi.encryptIndex(key); expect(encryptIndex).toHaveBeenCalled(); done(); }); it('should exist encrypted file', function (done) { setTimeout(function () { expect(fs.existsSync(path.join(userConfigDir, 'search_index_12345678910112.enc'))).toBe(true); expect(fs.existsSync(path.join(userConfigDir, 'search_index_12345678910112.tar.lz4'))).toBe(false); done(); }, 3000); }); }); describe('Test for latest timestamp', function () { it('should get the latest timestamp', function (done) { const getLatestMessageTimestamp = jest.spyOn(SearchApi, 'getLatestMessageTimestamp'); SearchApi.getLatestMessageTimestamp().then(function (res) { expect(res).toEqual(currentDate.toString()); expect(getLatestMessageTimestamp).toHaveBeenCalled(); done(); }); }); it('should not get the latest timestamp', function (done) { const getLatestMessageTimestamp = jest.spyOn(SearchApi, 'getLatestMessageTimestamp'); SearchApi.isInitialized = false; SearchApi.getLatestMessageTimestamp().catch(function (err) { expect(err).toEqual(new Error('Not initialized')); expect(getLatestMessageTimestamp).toHaveBeenCalled(); SearchApi.isInitialized = true; done(); }); }); it('should not get the latest timestamp', function (done) { const getLatestMessageTimestamp = jest.spyOn(SearchApi, 'getLatestMessageTimestamp'); deleteIndexFolders(dataFolderPath); SearchApi.getLatestMessageTimestamp().catch(function (err) { expect(err).toEqual(new Error('Index folder does not exist.')); expect(getLatestMessageTimestamp).toHaveBeenCalled(); expect(getLatestMessageTimestamp).toHaveBeenCalledTimes(3); done(); }); }); }); describe('Test to decrypt the index', function () { it('should decrypt the index', function (done) { setTimeout(function () { const decryptAndInit = jest.spyOn(SearchApi, 'decryptAndInit'); SearchApi.decryptAndInit(); expect(decryptAndInit).toHaveBeenCalled(); done(); }, 3000); }); it('should get message from the decrypted index', function (done) { setTimeout(function () { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); let endTime = new Date().getTime(); let startTime = new Date().getTime() - (4 * 31 * 24 * 60 * 60 * 1000); SearchApi.searchQuery('it works', [], [], '', startTime.toString(), endTime.toString(), '0', 0.2, 0.1).then(function (res) { expect(res.messages.length).toEqual(3); expect(searchQuery).toHaveBeenCalled(); done() }); }, 3000) }); }); describe('Test for search functions', function () { it('should search fail isInitialized is false', function (done) { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); SearchApi.isInitialized = false; SearchApi.searchQuery('it works', [], [], '', '', '', 25, 0, 0).catch(function (err) { expect(err).toEqual(new Error('Library not initialized')); expect(searchQuery).toHaveBeenCalled(); SearchApi.isInitialized = true; done(); }); }); it('should filter search limit ', function (done) { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); SearchApi.searchQuery('works', [], [], '', '', '', 2, 0, 0).then(function (res) { expect(res.messages.length).toBe(2); expect(searchQuery).toHaveBeenCalledTimes(7); expect(searchQuery).toHaveBeenCalled(); done(); }); }); it('should search fails index folder not fund', function (done) { deleteIndexFolders(dataFolderPath); setTimeout(function () { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); SearchApi.searchQuery('it works', [], [], '', '', '', 25, 0, 0).catch(function (err) { expect(err).toEqual(new Error('Index folder does not exist.')); expect(searchQuery).toHaveBeenCalledTimes(8); expect(searchQuery).toHaveBeenCalled(); SearchApi = undefined; const { Search } = require('../js/search/search.js'); SearchApi = new Search(userId, key); done(); }) }, 3000); }); it('should search fails query is undefined', function (done) { setTimeout(function () { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); expect(SearchApi.isInitialized).toBe(true); SearchApi.searchQuery(undefined, [], [], '', '', '', 25, 0, 0).catch(function (err) { expect(err).toEqual(new Error('Search query error')); expect(searchQuery).toHaveBeenCalled(); done(); }); }, 3000); }); it('should search for hashtag', function (done) { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); SearchApi.searchQuery('#123 "testing"', [], [], 'attachment', '', '', 25, 0, 0).then(function (res) { expect(res.messages.length).toEqual(0); expect(searchQuery).toHaveBeenCalled(); done(); }); }); it('should search for pdf', function (done) { const searchQuery = jest.spyOn(SearchApi, 'searchQuery'); SearchApi.searchQuery('', [], [], 'pdf', '', '', 25, 0, 0).then(function (res) { expect(res.messages.length).toEqual(0); expect(searchQuery).toHaveBeenCalled(); expect(searchQuery).toHaveBeenCalledTimes(3); done(); }); }); }); });