diff --git a/.eslintignore b/.eslintignore index 9584af704..bf3913992 100644 --- a/.eslintignore +++ b/.eslintignore @@ -1639,6 +1639,9 @@ packages/lib/services/searchengine/SearchEngineUtils.js.map packages/lib/services/searchengine/SearchEngineUtils.test.d.ts packages/lib/services/searchengine/SearchEngineUtils.test.js packages/lib/services/searchengine/SearchEngineUtils.test.js.map +packages/lib/services/searchengine/SearchFilter.test.d.ts +packages/lib/services/searchengine/SearchFilter.test.js +packages/lib/services/searchengine/SearchFilter.test.js.map packages/lib/services/searchengine/filterParser.d.ts packages/lib/services/searchengine/filterParser.js packages/lib/services/searchengine/filterParser.js.map diff --git a/.gitignore b/.gitignore index 02be4abce..abdb4efbb 100644 --- a/.gitignore +++ b/.gitignore @@ -1629,6 +1629,9 @@ packages/lib/services/searchengine/SearchEngineUtils.js.map packages/lib/services/searchengine/SearchEngineUtils.test.d.ts packages/lib/services/searchengine/SearchEngineUtils.test.js packages/lib/services/searchengine/SearchEngineUtils.test.js.map +packages/lib/services/searchengine/SearchFilter.test.d.ts +packages/lib/services/searchengine/SearchFilter.test.js +packages/lib/services/searchengine/SearchFilter.test.js.map packages/lib/services/searchengine/filterParser.d.ts packages/lib/services/searchengine/filterParser.js packages/lib/services/searchengine/filterParser.js.map diff --git a/packages/lib/services/searchengine/SearchFilter.test.js b/packages/lib/services/searchengine/SearchFilter.test.ts similarity index 85% rename from packages/lib/services/searchengine/SearchFilter.test.js rename to packages/lib/services/searchengine/SearchFilter.test.ts index 9a75f46c8..63e360659 100644 --- a/packages/lib/services/searchengine/SearchFilter.test.js +++ b/packages/lib/services/searchengine/SearchFilter.test.ts @@ -1,18 +1,19 @@ -/* eslint-disable no-unused-vars, @typescript-eslint/no-unused-vars, prefer-const */ +/* @typescript-eslint/prefer-const */ -const time = require('../../time').default; -const { setupDatabaseAndSynchronizer, supportDir, db, createNTestNotes, switchClient } = require('../../testing/test-utils.js'); -const SearchEngine = require('../../services/searchengine/SearchEngine').default; -const Note = require('../../models/Note').default; -const Folder = require('../../models/Folder').default; -const Tag = require('../../models/Tag').default; -const shim = require('../../shim').default; -const ResourceService = require('../../services/ResourceService').default; +import time from '@joplin/lib/time'; +import { setupDatabaseAndSynchronizer, supportDir, db, createNTestNotes, switchClient } from '@joplin/lib/testing//test-utils'; +import SearchEngine from '@joplin/lib/services/searchengine/SearchEngine'; +import Note from '@joplin/lib/models/Note'; +import Folder from '@joplin/lib/models/Folder'; +import Tag from '@joplin/lib/models/Tag'; +import shim from '@joplin/lib/shim'; +import ResourceService from '@joplin/lib/services/ResourceService'; +import { NoteEntity } from '@joplin/lib/services/database/types'; -let engine = null; +let engine: any = null; -const ids = (array) => array.map(a => a.id); +const ids = (array: NoteEntity[]) => array.map(a => a.id); describe('services_SearchFilter', function() { beforeEach(async (done) => { @@ -67,25 +68,59 @@ describe('services_SearchFilter', function() { for (const searchType of [SearchEngine.SEARCH_TYPE_FTS, SearchEngine.SEARCH_TYPE_NONLATIN_SCRIPT]) { describe(`search type ${searchType}`, () => { - it('should return note matching title', (async () => { + it('Check case insensitivity for filter keywords', (async () => { let rows; - const n1 = await Note.save({ title: 'abcd', body: 'body 1' }); - const n2 = await Note.save({ title: 'efgh', body: 'body 2' }); + const notebook1 = await Folder.save({ title: 'folderA' }); + const notebook2 = await Folder.save({ title: 'folderB' }); + const note1 = await Note.save({ title: 'Note1', body: 'obelix', parent_id: notebook1.id }); + const note2 = await Note.save({ title: 'Note2', body: 'asterix', parent_id: notebook2.id }); + const note3 = await Note.save({ title: 'Note3', body: 'rom', parent_id: notebook1.id }); + + await Tag.setNoteTagsByTitles(note1.id, ['tag1', 'tag2']); + await Tag.setNoteTagsByTitles(note2.id, ['tag2', 'tag3']); + await Tag.setNoteTagsByTitles(note3.id, ['tag3', 'tag4', 'space travel']); await engine.syncTables(); - rows = await engine.search('title: abcd', { searchType }); + + const testCases = [ + { searchString: 'tag:tag2', expectedResults: 2, expectedtNoteIds: [note1.id, note2.id] }, + { searchString: 'tAg:tag2', expectedResults: 2, expectedtNoteIds: [note1.id, note2.id] }, + { searchString: 'Tag:tag2', expectedResults: 2, expectedtNoteIds: [note1.id, note2.id] }, + { searchString: '-tag:tag2', expectedResults: 1, expectedtNoteIds: [note3.id] }, + { searchString: '-Tag:tag2', expectedResults: 1, expectedtNoteIds: [note3.id] }, + { searchString: 'title:Note1', expectedResults: 1, expectedtNoteIds: [note1.id] }, + { searchString: 'Title:Note1', expectedResults: 1, expectedtNoteIds: [note1.id] }, + { searchString: 'Any:1 -tag:tag1 -notebook:folderB', expectedResults: 1, expectedtNoteIds: [note3.id] }, + { searchString: 'notebook:folderA', expectedResults: 2, expectedtNoteIds: [note1.id, note3.id] }, + { searchString: 'notebooK:folderA', expectedResults: 2, expectedtNoteIds: [note1.id, note3.id] }, + ]; + + for (const testCase of testCases) { + rows = await engine.search(testCase.searchString, { searchType }); + expect(rows.length).toBe(testCase.expectedResults); + for (const expectedNoteId of testCase.expectedtNoteIds) { + expect(ids(rows)).toContain(expectedNoteId); + } + } + })); + + it('should return note matching title', (async () => { + const n1 = await Note.save({ title: 'abcd', body: 'body 1' }); + await Note.save({ title: 'efgh', body: 'body 2' }); + + await engine.syncTables(); + const rows = await engine.search('title: abcd', { searchType }); expect(rows.length).toBe(1); expect(rows[0].id).toBe(n1.id); })); it('should return note matching negated title', (async () => { - let rows; - const n1 = await Note.save({ title: 'abcd', body: 'body 1' }); + await Note.save({ title: 'abcd', body: 'body 1' }); const n2 = await Note.save({ title: 'efgh', body: 'body 2' }); await engine.syncTables(); - rows = await engine.search('-title: abcd', { searchType }); + const rows = await engine.search('-title: abcd', { searchType }); expect(rows.length).toBe(1); @@ -93,12 +128,11 @@ describe('services_SearchFilter', function() { })); it('should return note matching body', (async () => { - let rows; const n1 = await Note.save({ title: 'abcd', body: 'body1' }); - const n2 = await Note.save({ title: 'efgh', body: 'body2' }); + await Note.save({ title: 'efgh', body: 'body2' }); await engine.syncTables(); - rows = await engine.search('body: body1', { searchType }); + const rows = await engine.search('body: body1', { searchType }); expect(rows.length).toBe(1); @@ -106,36 +140,33 @@ describe('services_SearchFilter', function() { })); it('should return note matching negated body', (async () => { - let rows; - const n1 = await Note.save({ title: 'abcd', body: 'body1' }); + await Note.save({ title: 'abcd', body: 'body1' }); const n2 = await Note.save({ title: 'efgh', body: 'body2' }); await engine.syncTables(); - rows = await engine.search('-body: body1', { searchType }); + const rows = await engine.search('-body: body1', { searchType }); expect(rows.length).toBe(1); expect(rows[0].id).toBe(n2.id); })); it('should return note matching title containing multiple words', (async () => { - let rows; const n1 = await Note.save({ title: 'abcd xyz', body: 'body1' }); - const n2 = await Note.save({ title: 'efgh ijk', body: 'body2' }); + await Note.save({ title: 'efgh ijk', body: 'body2' }); await engine.syncTables(); - rows = await engine.search('title: "abcd xyz"', { searchType }); + const rows = await engine.search('title: "abcd xyz"', { searchType }); expect(rows.length).toBe(1); expect(rows[0].id).toBe(n1.id); })); it('should return note matching body containing multiple words', (async () => { - let rows; - const n1 = await Note.save({ title: 'abcd', body: 'ho ho ho' }); + await Note.save({ title: 'abcd', body: 'ho ho ho' }); const n2 = await Note.save({ title: 'efgh', body: 'foo bar' }); await engine.syncTables(); - rows = await engine.search('body: "foo bar"', { searchType }); + const rows = await engine.search('body: "foo bar"', { searchType }); expect(rows.length).toBe(1); expect(rows[0].id).toBe(n2.id); @@ -143,7 +174,7 @@ describe('services_SearchFilter', function() { it('should return note matching title AND body', (async () => { let rows; - const n1 = await Note.save({ title: 'abcd', body: 'ho ho ho' }); + await Note.save({ title: 'abcd', body: 'ho ho ho' }); const n2 = await Note.save({ title: 'efgh', body: 'foo bar' }); await engine.syncTables(); @@ -163,8 +194,8 @@ describe('services_SearchFilter', function() { await engine.syncTables(); rows = await engine.search('any:1 title: abcd body: "foo bar"', { searchType }); expect(rows.length).toBe(2); - expect(rows.map(r=>r.id)).toContain(n1.id); - expect(rows.map(r=>r.id)).toContain(n2.id); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); rows = await engine.search('any:1 title: wxyz body: "blah blah"', { searchType }); expect(rows.length).toBe(0); @@ -181,12 +212,12 @@ describe('services_SearchFilter', function() { // Note: This is NOT saying to match notes containing foo bar in title/body rows = await engine.search('foo bar', { searchType }); expect(rows.length).toBe(2); - expect(rows.map(r=>r.id)).toContain(n1.id); - expect(rows.map(r=>r.id)).toContain(n2.id); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); rows = await engine.search('foo -bar', { searchType }); expect(rows.length).toBe(1); - expect(rows.map(r=>r.id)).toContain(n3.id); + expect(ids(rows)).toContain(n3.id); rows = await engine.search('foo efgh', { searchType }); expect(rows.length).toBe(1); @@ -197,65 +228,60 @@ describe('services_SearchFilter', function() { })); it('should return notes matching any negated text', (async () => { - let rows; const n1 = await Note.save({ title: 'abc', body: 'def' }); const n2 = await Note.save({ title: 'def', body: 'ghi' }); const n3 = await Note.save({ title: 'ghi', body: 'jkl' }); await engine.syncTables(); - rows = await engine.search('any:1 -abc -ghi', { searchType }); + const rows = await engine.search('any:1 -abc -ghi', { searchType }); expect(rows.length).toBe(3); - expect(rows.map(r=>r.id)).toContain(n1.id); - expect(rows.map(r=>r.id)).toContain(n2.id); - expect(rows.map(r=>r.id)).toContain(n3.id); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); })); it('should return notes matching any negated title', (async () => { - let rows; const n1 = await Note.save({ title: 'abc', body: 'def' }); const n2 = await Note.save({ title: 'def', body: 'ghi' }); const n3 = await Note.save({ title: 'ghi', body: 'jkl' }); await engine.syncTables(); - rows = await engine.search('any:1 -title:abc -title:ghi', { searchType }); + const rows = await engine.search('any:1 -title:abc -title:ghi', { searchType }); expect(rows.length).toBe(3); - expect(rows.map(r=>r.id)).toContain(n1.id); - expect(rows.map(r=>r.id)).toContain(n2.id); - expect(rows.map(r=>r.id)).toContain(n3.id); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); })); it('should return notes matching any negated body', (async () => { - let rows; const n1 = await Note.save({ title: 'abc', body: 'def' }); const n2 = await Note.save({ title: 'def', body: 'ghi' }); const n3 = await Note.save({ title: 'ghi', body: 'jkl' }); await engine.syncTables(); - rows = await engine.search('any:1 -body:xyz -body:ghi', { searchType }); + const rows = await engine.search('any:1 -body:xyz -body:ghi', { searchType }); expect(rows.length).toBe(3); - expect(rows.map(r=>r.id)).toContain(n1.id); - expect(rows.map(r=>r.id)).toContain(n2.id); - expect(rows.map(r=>r.id)).toContain(n3.id); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); })); it('should support phrase search', (async () => { - let rows; const n1 = await Note.save({ title: 'foo beef', body: 'bar dog' }); - const n2 = await Note.save({ title: 'bar efgh', body: 'foo dog' }); + await Note.save({ title: 'bar efgh', body: 'foo dog' }); await engine.syncTables(); - rows = await engine.search('"bar dog"', { searchType }); + const rows = await engine.search('"bar dog"', { searchType }); expect(rows.length).toBe(1); expect(rows[0].id).toBe(n1.id); })); it('should support prefix search', (async () => { - let rows; const n1 = await Note.save({ title: 'foo beef', body: 'bar dog' }); const n2 = await Note.save({ title: 'bar efgh', body: 'foo dog' }); await engine.syncTables(); - rows = await engine.search('"bar*"', { searchType }); + const rows = await engine.search('"bar*"', { searchType }); expect(rows.length).toBe(2); expect(ids(rows)).toContain(n1.id); expect(ids(rows)).toContain(n2.id); @@ -338,38 +364,35 @@ describe('services_SearchFilter', function() { })); it('should support filtering by notebook', (async () => { - let rows; const folder0 = await Folder.save({ title: 'notebook0' }); const folder1 = await Folder.save({ title: 'notebook1' }); const notes0 = await createNTestNotes(5, folder0); - const notes1 = await createNTestNotes(5, folder1); + await createNTestNotes(5, folder1); await engine.syncTables(); - rows = await engine.search('notebook:notebook0', { searchType }); + const rows = await engine.search('notebook:notebook0', { searchType }); expect(rows.length).toBe(5); expect(ids(rows).sort()).toEqual(ids(notes0).sort()); })); it('should support filtering by nested notebook', (async () => { - let rows; const folder0 = await Folder.save({ title: 'notebook0' }); const folder00 = await Folder.save({ title: 'notebook00', parent_id: folder0.id }); const folder1 = await Folder.save({ title: 'notebook1' }); const notes0 = await createNTestNotes(5, folder0); const notes00 = await createNTestNotes(5, folder00); - const notes1 = await createNTestNotes(5, folder1); + await createNTestNotes(5, folder1); await engine.syncTables(); - rows = await engine.search('notebook:notebook0', { searchType }); + const rows = await engine.search('notebook:notebook0', { searchType }); expect(rows.length).toBe(10); expect(ids(rows).sort()).toEqual(ids(notes0.concat(notes00)).sort()); })); it('should support filtering by multiple notebooks', (async () => { - let rows; const folder0 = await Folder.save({ title: 'notebook0' }); const folder00 = await Folder.save({ title: 'notebook00', parent_id: folder0.id }); const folder1 = await Folder.save({ title: 'notebook1' }); @@ -377,11 +400,11 @@ describe('services_SearchFilter', function() { const notes0 = await createNTestNotes(5, folder0); const notes00 = await createNTestNotes(5, folder00); const notes1 = await createNTestNotes(5, folder1); - const notes2 = await createNTestNotes(5, folder2); + await createNTestNotes(5, folder2); await engine.syncTables(); - rows = await engine.search('notebook:notebook0 notebook:notebook1', { searchType }); + const rows = await engine.search('notebook:notebook0 notebook:notebook1', { searchType }); expect(rows.length).toBe(15); expect(ids(rows).sort()).toEqual(ids(notes0).concat(ids(notes00).concat(ids(notes1))).sort()); })); @@ -610,7 +633,7 @@ describe('services_SearchFilter', function() { let rows; const t1 = await Note.save({ title: 'This is a ', body: 'todo', is_todo: 1 }); const t2 = await Note.save({ title: 'This is another', body: 'todo but completed', is_todo: 1, todo_completed: 1590085027710 }); - const t3 = await Note.save({ title: 'This is NOT a ', body: 'todo' }); + await Note.save({ title: 'This is NOT a ', body: 'todo' }); await engine.syncTables(); @@ -634,14 +657,13 @@ describe('services_SearchFilter', function() { })); it('should support filtering by type note', (async () => { - let rows; - const t1 = await Note.save({ title: 'This is a ', body: 'todo', is_todo: 1 }); - const t2 = await Note.save({ title: 'This is another', body: 'todo but completed', is_todo: 1, todo_completed: 1590085027710 }); + await Note.save({ title: 'This is a ', body: 'todo', is_todo: 1 }); + await Note.save({ title: 'This is another', body: 'todo but completed', is_todo: 1, todo_completed: 1590085027710 }); const t3 = await Note.save({ title: 'This is NOT a ', body: 'todo' }); await engine.syncTables(); - rows = await engine.search('type:note', { searchType }); + const rows = await engine.search('type:note', { searchType }); expect(rows.length).toBe(1); expect(ids(rows)).toContain(t3.id); })); @@ -650,7 +672,7 @@ describe('services_SearchFilter', function() { let rows; const toDo1 = await Note.save({ title: 'ToDo 1', body: 'todo', is_todo: 1, todo_due: Date.parse('2021-04-27') }); const toDo2 = await Note.save({ title: 'ToDo 2', body: 'todo', is_todo: 1, todo_due: Date.parse('2021-03-17') }); - const note1 = await Note.save({ title: 'Note 1', body: 'Note' }); + await Note.save({ title: 'Note 1', body: 'Note' }); await engine.syncTables(); @@ -864,8 +886,8 @@ describe('services_SearchFilter', function() { const subFolder = await Folder.save({ title: 'child', parent_id: parentFolder.id }); - const n3 = await Note.save({ title: 'task3', body: 'baz', parent_id: subFolder.id }); - const n4 = await Note.save({ title: 'task4', body: 'blah', parent_id: subFolder.id }); + await Note.save({ title: 'task3', body: 'baz', parent_id: subFolder.id }); + await Note.save({ title: 'task4', body: 'blah', parent_id: subFolder.id }); await engine.syncTables(); @@ -886,20 +908,20 @@ describe('services_SearchFilter', function() { rows = await engine.search(`id:${note1.id}`, { searchType }); expect(rows.length).toBe(1); - expect(rows.map(r=>r.id)).toContain(note1.id); + expect(ids(rows)).toContain(note1.id); rows = await engine.search(`any:1 id:${note1.id} id:${note2.id}`, { searchType }); expect(rows.length).toBe(2); - expect(rows.map(r=>r.id)).toContain(note1.id); - expect(rows.map(r=>r.id)).toContain(note2.id); + expect(ids(rows)).toContain(note1.id); + expect(ids(rows)).toContain(note2.id); rows = await engine.search(`any:0 id:${note1.id} id:${note2.id}`, { searchType }); expect(rows.length).toBe(0); rows = await engine.search(`-id:${note2.id}`, { searchType }); expect(rows.length).toBe(2); - expect(rows.map(r=>r.id)).toContain(note1.id); - expect(rows.map(r=>r.id)).toContain(note3.id); + expect(ids(rows)).toContain(note1.id); + expect(ids(rows)).toContain(note3.id); })); }); diff --git a/packages/lib/services/searchengine/filterParser.ts b/packages/lib/services/searchengine/filterParser.ts index 82a0aa558..24ca48347 100644 --- a/packages/lib/services/searchengine/filterParser.ts +++ b/packages/lib/services/searchengine/filterParser.ts @@ -54,8 +54,8 @@ const getTerms = (query: string, validFilters: Set): Term[] => { } if (c === ':' && !inQuote && !inTerm && - (validFilters.has(currentTerm) || currentTerm[0] === '-' && validFilters.has(currentTerm.substr(1, currentTerm.length)))) { - currentCol = currentTerm; + (validFilters.has(currentTerm.toLowerCase()) || currentTerm[0] === '-' && validFilters.has(currentTerm.toLowerCase().substr(1, currentTerm.length)))) { + currentCol = currentTerm.toLowerCase(); currentTerm = ''; inTerm = true; // to ignore any other ':' before a space eg.'sourceurl:https://www.google.com' continue;