From f99f3f8a6d15f91957cd2b24cb0fe7d96835cf35 Mon Sep 17 00:00:00 2001 From: Naveen M V <30305957+naviji@users.noreply.github.com> Date: Sat, 8 Aug 2020 04:43:21 +0530 Subject: [PATCH 1/4] All: Resolves #1877: Add search filters (#3213) --- .eslintignore | 2 + .gitignore | 2 + CliClient/tests/filterParser.js | 155 ++++ CliClient/tests/models_ItemChange.js | 2 +- CliClient/tests/services_ResourceService.js | 2 +- CliClient/tests/services_SearchEngine.js | 3 +- CliClient/tests/services_SearchFilter.js | 752 ++++++++++++++++++ CliClient/tests/support/welcome.pdf | Bin 0 -> 6736 bytes CliClient/tests/timeUtils.js | 66 ++ .../gui/NoteEditor/utils/useSearchMarkers.ts | 2 +- ElectronClient/gui/NoteList/NoteList.jsx | 2 +- ElectronClient/plugins/GotoAnything.jsx | 2 +- ReactNativeClient/lib/BaseApplication.js | 4 +- .../lib/components/screens/config.js | 2 +- .../lib/components/screens/note.js | 2 +- .../lib/components/screens/search.js | 2 +- ReactNativeClient/lib/joplin-database.js | 89 ++- ReactNativeClient/lib/migrations/33.js | 9 + ReactNativeClient/lib/models/Migration.js | 1 + .../lib/services/ResourceService.js | 2 +- ReactNativeClient/lib/services/rest/Api.js | 2 +- .../{ => searchengine}/SearchEngine.js | 116 ++- .../{ => searchengine}/SearchEngineUtils.js | 10 +- .../lib/services/searchengine/filterParser.ts | 133 ++++ .../lib/services/searchengine/queryBuilder.ts | 428 ++++++++++ ReactNativeClient/lib/time-utils.js | 11 + ReactNativeClient/root.js | 2 +- 27 files changed, 1715 insertions(+), 88 deletions(-) create mode 100644 CliClient/tests/filterParser.js create mode 100644 CliClient/tests/services_SearchFilter.js create mode 100644 CliClient/tests/support/welcome.pdf create mode 100644 CliClient/tests/timeUtils.js create mode 100644 ReactNativeClient/lib/migrations/33.js rename ReactNativeClient/lib/services/{ => searchengine}/SearchEngine.js (83%) rename ReactNativeClient/lib/services/{ => searchengine}/SearchEngineUtils.js (87%) create mode 100644 ReactNativeClient/lib/services/searchengine/filterParser.ts create mode 100644 ReactNativeClient/lib/services/searchengine/queryBuilder.ts diff --git a/.eslintignore b/.eslintignore index 167257849..e5064b015 100644 --- a/.eslintignore +++ b/.eslintignore @@ -164,6 +164,8 @@ ReactNativeClient/lib/services/ResourceEditWatcher/index.js ReactNativeClient/lib/services/ResourceEditWatcher/reducer.js ReactNativeClient/lib/services/rest/actionApi.desktop.js ReactNativeClient/lib/services/rest/errors.js +ReactNativeClient/lib/services/searchengine/filterParser.js +ReactNativeClient/lib/services/searchengine/queryBuilder.js ReactNativeClient/lib/services/SettingUtils.js ReactNativeClient/lib/services/synchronizer/gui/useSyncTargetUpgrade.js ReactNativeClient/lib/services/synchronizer/LockHandler.js diff --git a/.gitignore b/.gitignore index a6b60c065..01e8555ca 100644 --- a/.gitignore +++ b/.gitignore @@ -155,6 +155,8 @@ ReactNativeClient/lib/services/ResourceEditWatcher/index.js ReactNativeClient/lib/services/ResourceEditWatcher/reducer.js ReactNativeClient/lib/services/rest/actionApi.desktop.js ReactNativeClient/lib/services/rest/errors.js +ReactNativeClient/lib/services/searchengine/filterParser.js +ReactNativeClient/lib/services/searchengine/queryBuilder.js ReactNativeClient/lib/services/SettingUtils.js ReactNativeClient/lib/services/synchronizer/gui/useSyncTargetUpgrade.js ReactNativeClient/lib/services/synchronizer/LockHandler.js diff --git a/CliClient/tests/filterParser.js b/CliClient/tests/filterParser.js new file mode 100644 index 000000000..d6d4bbc9b --- /dev/null +++ b/CliClient/tests/filterParser.js @@ -0,0 +1,155 @@ +/* eslint-disable no-unused-vars */ + +require('app-module-path').addPath(__dirname); +const filterParser = require('lib/services/searchengine/filterParser.js').default; +// import filterParser from 'lib/services/searchengine/filterParser.js'; + +const makeTerm = (name, value, negated) => { return { name, value, negated }; }; +describe('filterParser should be correct filter for keyword', () => { + it('title', () => { + const searchString = 'title: something'; + expect(filterParser(searchString)).toContain(makeTerm('title', 'something', false)); + }); + + it('negated title', () => { + const searchString = '-title: something'; + expect(filterParser(searchString)).toContain(makeTerm('title', 'something', true)); + }); + + it('body', () => { + const searchString = 'body:something'; + expect(filterParser(searchString)).toContain(makeTerm('body', 'something', false)); + }); + + it('negated body', () => { + const searchString = '-body:something'; + expect(filterParser(searchString)).toContain(makeTerm('body', 'something', true)); + }); + + it('title and body', () => { + const searchString = 'title:testTitle body:testBody'; + expect(filterParser(searchString)).toContain(makeTerm('title', 'testTitle', false)); + expect(filterParser(searchString)).toContain(makeTerm('body', 'testBody', false)); + }); + + it('title with multiple words', () => { + const searchString = 'title:"word1 word2" body:testBody'; + expect(filterParser(searchString)).toContain(makeTerm('title', 'word1', false)); + expect(filterParser(searchString)).toContain(makeTerm('title', 'word2', false)); + expect(filterParser(searchString)).toContain(makeTerm('body', 'testBody', false)); + }); + + it('body with multiple words', () => { + const searchString = 'title:testTitle body:"word1 word2"'; + expect(filterParser(searchString)).toContain(makeTerm('title', 'testTitle', false)); + expect(filterParser(searchString)).toContain(makeTerm('body', 'word1', false)); + expect(filterParser(searchString)).toContain(makeTerm('body', 'word2', false)); + }); + + it('single word text', () => { + const searchString = 'joplin'; + expect(filterParser(searchString)).toContain(makeTerm('text', '"joplin"', false)); + }); + + it('multi word text', () => { + const searchString = 'scott joplin'; + expect(filterParser(searchString)).toContain(makeTerm('text', '"scott"', false)); + expect(filterParser(searchString)).toContain(makeTerm('text', '"joplin"', false)); + }); + + it('negated word text', () => { + const searchString = 'scott -joplin'; + expect(filterParser(searchString)).toContain(makeTerm('text', '"scott"', false)); + expect(filterParser(searchString)).toContain(makeTerm('text', '"joplin"', true)); + }); + + it('phrase text search', () => { + const searchString = '"scott joplin"'; + expect(filterParser(searchString)).toContain(makeTerm('text', '"scott joplin"', false)); + }); + + it('multi word body', () => { + const searchString = 'body:"foo bar"'; + expect(filterParser(searchString)).toContain(makeTerm('body', 'foo', false)); + expect(filterParser(searchString)).toContain(makeTerm('body', 'bar', false)); + }); + + it('negated tag queries', () => { + const searchString = '-tag:mozart'; + expect(filterParser(searchString)).toContain(makeTerm('tag', 'mozart', true)); + }); + + + it('created after', () => { + const searchString = 'created:20151218'; // YYYYMMDD + expect(filterParser(searchString)).toContain(makeTerm('created', '20151218', false)); + }); + + it('created before', () => { + const searchString = '-created:20151218'; // YYYYMMDD + expect(filterParser(searchString)).toContain(makeTerm('created', '20151218', true)); + }); + + it('any', () => { + const searchString = 'any:1 tag:123'; + expect(filterParser(searchString)).toContain(makeTerm('any', '1', false)); + expect(filterParser(searchString)).toContain(makeTerm('tag', '123', false)); + }); + + it('wildcard tags', () => { + let searchString = 'tag:*'; + expect(filterParser(searchString)).toContain(makeTerm('tag', '%', false)); + + searchString = '-tag:*'; + expect(filterParser(searchString)).toContain(makeTerm('tag', '%', true)); + + searchString = 'tag:bl*sphemy'; + expect(filterParser(searchString)).toContain(makeTerm('tag', 'bl%sphemy', false)); + }); + + it('wildcard notebooks', () => { + const searchString = 'notebook:my*notebook'; + expect(filterParser(searchString)).toContain(makeTerm('notebook', 'my%notebook', false)); + }); + + it('wildcard MIME types', () => { + const searchString = 'resource:image/*'; + expect(filterParser(searchString)).toContain(makeTerm('resource', 'image/%', false)); + }); + + it('sourceurl', () => { + let searchString = 'sourceurl:https://www.google.com'; + expect(filterParser(searchString)).toContain(makeTerm('sourceurl', 'https://www.google.com', false)); + + searchString = 'sourceurl:https://www.google.com -sourceurl:https://www.facebook.com'; + expect(filterParser(searchString)).toContain(makeTerm('sourceurl', 'https://www.google.com', false)); + expect(filterParser(searchString)).toContain(makeTerm('sourceurl', 'https://www.facebook.com', true)); + }); + + it('handle invalid filters', () => { + let searchString = 'titletitle:123'; + expect(() => filterParser(searchString)).toThrow(new Error('Invalid filter: titletitle')); + + searchString = 'invalid:abc'; + expect(() => filterParser(searchString)).toThrow(new Error('Invalid filter: invalid')); + + searchString = ':abc'; + expect(() => filterParser(searchString)).toThrow(new Error('Invalid filter: ')); + + searchString = 'type:blah'; + expect(() => filterParser(searchString)).toThrow(new Error('The value of filter "type" must be "note" or "todo"')); + + searchString = '-type:note'; + expect(() => filterParser(searchString)).toThrow(new Error('type can\'t be negated')); + + searchString = 'iscompleted:blah'; + expect(() => filterParser(searchString)).toThrow(new Error('The value of filter "iscompleted" must be "1" or "0"')); + + searchString = '-notebook:n1'; + expect(() => filterParser(searchString)).toThrow(new Error('notebook can\'t be negated')); + + searchString = '-iscompleted:1'; + expect(() => filterParser(searchString)).toThrow(new Error('iscompleted can\'t be negated')); + + }); +}); diff --git a/CliClient/tests/models_ItemChange.js b/CliClient/tests/models_ItemChange.js index 66a3a212b..c1ac6c8eb 100644 --- a/CliClient/tests/models_ItemChange.js +++ b/CliClient/tests/models_ItemChange.js @@ -4,7 +4,7 @@ require('app-module-path').addPath(__dirname); const { time } = require('lib/time-utils.js'); const { asyncTest, fileContentEqual, revisionService, setupDatabase, setupDatabaseAndSynchronizer, db, synchronizer, fileApi, sleep, clearDatabase, switchClient, syncTargetId, objectsEqual, checkThrowAsync } = require('test-utils.js'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const ResourceService = require('lib/services/ResourceService'); const ItemChangeUtils = require('lib/services/ItemChangeUtils'); const Note = require('lib/models/Note'); diff --git a/CliClient/tests/services_ResourceService.js b/CliClient/tests/services_ResourceService.js index 149fa8273..9a0072f18 100644 --- a/CliClient/tests/services_ResourceService.js +++ b/CliClient/tests/services_ResourceService.js @@ -17,7 +17,7 @@ const fs = require('fs-extra'); const ArrayUtils = require('lib/ArrayUtils'); const ObjectUtils = require('lib/ObjectUtils'); const { shim } = require('lib/shim.js'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); process.on('unhandledRejection', (reason, p) => { console.log('Unhandled Rejection at: Promise', p, 'reason:', reason); diff --git a/CliClient/tests/services_SearchEngine.js b/CliClient/tests/services_SearchEngine.js index 59c45682f..bca863670 100644 --- a/CliClient/tests/services_SearchEngine.js +++ b/CliClient/tests/services_SearchEngine.js @@ -5,7 +5,7 @@ require('app-module-path').addPath(__dirname); const { time } = require('lib/time-utils.js'); const { fileContentEqual, setupDatabase, setupDatabaseAndSynchronizer, asyncTest, db, synchronizer, fileApi, sleep, clearDatabase, switchClient, syncTargetId, objectsEqual, checkThrowAsync } = require('test-utils.js'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const Note = require('lib/models/Note'); const ItemChange = require('lib/models/ItemChange'); const Setting = require('lib/models/Setting'); @@ -389,5 +389,4 @@ describe('services_SearchEngine', function() { expect((await engine.search('"- [ ]"', { searchType: SearchEngine.SEARCH_TYPE_BASIC })).length).toBe(1); expect((await engine.search('"[ ]"', { searchType: SearchEngine.SEARCH_TYPE_BASIC })).length).toBe(2); })); - }); diff --git a/CliClient/tests/services_SearchFilter.js b/CliClient/tests/services_SearchFilter.js new file mode 100644 index 000000000..5a698e6f6 --- /dev/null +++ b/CliClient/tests/services_SearchFilter.js @@ -0,0 +1,752 @@ +/* eslint-disable no-unused-vars */ +/* eslint prefer-const: 0*/ + +require('app-module-path').addPath(__dirname); + +const { time } = require('lib/time-utils.js'); +const { fileContentEqual, setupDatabase, setupDatabaseAndSynchronizer, asyncTest, db, synchronizer, fileApi, sleep, createNTestNotes, switchClient, createNTestFolders } = require('test-utils.js'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); +const Note = require('lib/models/Note'); +const Folder = require('lib/models/Folder'); +const Tag = require('lib/models/Tag'); +const ItemChange = require('lib/models/ItemChange'); +const Setting = require('lib/models/Setting'); +const Resource = require('lib/models/Resource.js'); +const { shim } = require('lib/shim'); +const ResourceService = require('lib/services/ResourceService.js'); + + +process.on('unhandledRejection', (reason, p) => { + console.log('Unhandled Rejection at: Promise', p, 'reason:', reason); +}); + +let engine = null; + +const ids = (array) => array.map(a => a.id); + +// For pretty printing. +// See https://stackoverflow.com/questions/23676459/karma-jasmine-pretty-printing-object-comparison/26324116 +// jasmine.pp = function(obj) { +// return JSON.stringify(obj, undefined, 2); +// }; + +describe('services_SearchFilter', function() { + beforeEach(async (done) => { + await setupDatabaseAndSynchronizer(1); + await switchClient(1); + + engine = new SearchEngine(); + engine.setDb(db()); + + done(); + }); + + + it('should return note matching title', asyncTest(async () => { + let rows; + const n1 = 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'); + + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n1.id); + })); + + it('should return note matching negated title', asyncTest(async () => { + let rows; + const n1 = 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'); + + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n2.id); + })); + + it('should return note matching body', asyncTest(async () => { + let rows; + const n1 = 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'); + + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n1.id); + })); + + it('should return note matching negated body', asyncTest(async () => { + let rows; + const n1 = 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'); + + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n2.id); + })); + + it('should return note matching title containing multiple words', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'abcd xyz', body: 'body1' }); + const n2 = await Note.save({ title: 'efgh ijk', body: 'body2' }); + + await engine.syncTables(); + rows = await engine.search('title: "abcd xyz"'); + + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n1.id); + })); + + it('should return note matching body containing multiple words', asyncTest(async () => { + let rows; + const n1 = 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"'); + + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n2.id); + })); + + it('should return note matching title AND body', asyncTest(async () => { + let rows; + const n1 = 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('title: efgh body: "foo bar"'); + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n2.id); + + rows = await engine.search('title: abcd body: "foo bar"'); + expect(rows.length).toBe(0); + })); + + it('should return note matching title OR body', asyncTest(async () => { + let rows; + const n1 = 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('any:1 title: abcd body: "foo bar"'); + expect(rows.length).toBe(2); + expect(rows.map(r=>r.id)).toContain(n1.id); + expect(rows.map(r=>r.id)).toContain(n2.id); + + rows = await engine.search('any:1 title: wxyz body: "blah blah"'); + expect(rows.length).toBe(0); + })); + + it('should return notes matching text', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'foo beef', body: 'dead bar' }); + const n2 = await Note.save({ title: 'bar efgh', body: 'foo dog' }); + const n3 = await Note.save({ title: 'foo ho', body: 'ho ho ho' }); + await engine.syncTables(); + + // Interpretation: Match with notes containing foo in title/body and bar in title/body + // Note: This is NOT saying to match notes containing foo bar in title/body + rows = await engine.search('foo bar'); + expect(rows.length).toBe(2); + expect(rows.map(r=>r.id)).toContain(n1.id); + expect(rows.map(r=>r.id)).toContain(n2.id); + + rows = await engine.search('foo -bar'); + expect(rows.length).toBe(1); + expect(rows.map(r=>r.id)).toContain(n3.id); + + rows = await engine.search('foo efgh'); + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n2.id); + + rows = await engine.search('zebra'); + expect(rows.length).toBe(0); + })); + + it('should return notes matching any negated text', asyncTest(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'); + 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); + })); + + it('should return notes matching any negated title', asyncTest(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'); + 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); + })); + + it('should return notes matching any negated body', asyncTest(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'); + 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); + })); + + it('should support phrase search', asyncTest(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 dog"'); + expect(rows.length).toBe(1); + expect(rows[0].id).toBe(n1.id); + })); + + it('should support prefix search', asyncTest(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*"'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + })); + + it('should support filtering by tags', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'But I would', body: 'walk 500 miles' }); + const n2 = await Note.save({ title: 'And I would', body: 'walk 500 more' }); + const n3 = await Note.save({ title: 'Just to be', body: 'the man who' }); + const n4 = await Note.save({ title: 'walked a thousand', body: 'miles to fall' }); + const n5 = await Note.save({ title: 'down at your', body: 'door' }); + + await Tag.setNoteTagsByTitles(n1.id, ['Da', 'da', 'lat', 'da']); + await Tag.setNoteTagsByTitles(n2.id, ['Da', 'da', 'lat', 'da']); + + await engine.syncTables(); + + rows = await engine.search('tag:*'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('-tag:*'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n3.id); + expect(ids(rows)).toContain(n4.id); + expect(ids(rows)).toContain(n5.id); + })); + + + it('should support filtering by tags', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'peace talks', body: 'battle ground' }); + const n2 = await Note.save({ title: 'mouse', body: 'mister' }); + const n3 = await Note.save({ title: 'dresden files', body: 'harry dresden' }); + + await Tag.setNoteTagsByTitles(n1.id, ['tag1', 'tag2']); + await Tag.setNoteTagsByTitles(n2.id, ['tag2', 'tag3']); + await Tag.setNoteTagsByTitles(n3.id, ['tag3', 'tag4']); + + await engine.syncTables(); + + rows = await engine.search('tag:tag2'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('tag:tag2 tag:tag3'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('any:1 tag:tag1 tag:tag2 tag:tag3'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('tag:tag2 tag:tag3 tag:tag4'); + expect(rows.length).toBe(0); + + rows = await engine.search('-tag:tag2'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('-tag:tag2 -tag:tag3'); + expect(rows.length).toBe(0); + + rows = await engine.search('-tag:tag2 -tag:tag3'); + expect(rows.length).toBe(0); + + rows = await engine.search('any:1 -tag:tag2 -tag:tag3'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n3.id); + })); + + it('should support filtering by notebook', asyncTest(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 engine.syncTables(); + + rows = await engine.search('notebook:notebook0'); + expect(rows.length).toBe(5); + expect(ids(rows).sort()).toEqual(ids(notes0).sort()); + + })); + + it('should support filtering by nested notebook', asyncTest(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 engine.syncTables(); + + rows = await engine.search('notebook:notebook0'); + expect(rows.length).toBe(10); + expect(ids(rows).sort()).toEqual(ids(notes0.concat(notes00)).sort()); + })); + + it('should support filtering by multiple notebooks', asyncTest(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 folder2 = await Folder.save({ title: 'notebook2' }); + 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 engine.syncTables(); + + rows = await engine.search('notebook:notebook0 notebook:notebook1'); + expect(rows.length).toBe(15); + expect(ids(rows).sort()).toEqual(ids(notes0).concat(ids(notes00).concat(ids(notes1))).sort()); + })); + + it('should support filtering by created date', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'I made this on', body: 'May 20 2020', user_created_time: Date.parse('2020-05-20') }); + const n2 = await Note.save({ title: 'I made this on', body: 'May 19 2020', user_created_time: Date.parse('2020-05-19') }); + const n3 = await Note.save({ title: 'I made this on', body: 'May 18 2020', user_created_time: Date.parse('2020-05-18') }); + + await engine.syncTables(); + + rows = await engine.search('created:20200520'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('created:20200519'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('-created:20200519'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n3.id); + + })); + + it('should support filtering by between two dates', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'January 01 2020', body: 'January 01 2020', user_created_time: Date.parse('2020-01-01') }); + const n2 = await Note.save({ title: 'February 15 2020', body: 'February 15 2020', user_created_time: Date.parse('2020-02-15') }); + const n3 = await Note.save({ title: 'March 25 2019', body: 'March 25 2019', user_created_time: Date.parse('2019-03-25') }); + const n4 = await Note.save({ title: 'March 01 2018', body: 'March 01 2018', user_created_time: Date.parse('2018-03-01') }); + + await engine.syncTables(); + + rows = await engine.search('created:20200101 -created:20200220'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('created:201901 -created:202002'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n3.id); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('created:2018 -created:2019'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n4.id); + })); + + it('should support filtering by created with smart value: day', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'I made this', body: 'today', user_created_time: parseInt(time.goBackInTime(Date.now(), 0, 'day'), 10) }); + const n2 = await Note.save({ title: 'I made this', body: 'yesterday', user_created_time: parseInt(time.goBackInTime(Date.now(), 1, 'day'), 10) }); + const n3 = await Note.save({ title: 'I made this', body: 'day before yesterday', user_created_time: parseInt(time.goBackInTime(Date.now(), 2, 'day'), 10) }); + + await engine.syncTables(); + + rows = await engine.search('created:day-0'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('created:day-1'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('created:day-2'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + })); + + it('should support filtering by created with smart value: week', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'I made this', body: 'this week', user_created_time: parseInt(time.goBackInTime(Date.now(), 0, 'week'), 10) }); + const n2 = await Note.save({ title: 'I made this', body: 'the week before', user_created_time: parseInt(time.goBackInTime(Date.now(), 1, 'week'), 10) }); + const n3 = await Note.save({ title: 'I made this', body: 'before before week', user_created_time: parseInt(time.goBackInTime(Date.now(), 2, 'week'), 10) }); + + await engine.syncTables(); + + rows = await engine.search('created:week-0'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('created:week-1'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('created:week-2'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + })); + + it('should support filtering by created with smart value: month', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'I made this', body: 'this month', user_created_time: parseInt(time.goBackInTime(Date.now(), 0, 'month'), 10) }); + const n2 = await Note.save({ title: 'I made this', body: 'the month before', user_created_time: parseInt(time.goBackInTime(Date.now(), 1, 'month'), 10) }); + const n3 = await Note.save({ title: 'I made this', body: 'before before month', user_created_time: parseInt(time.goBackInTime(Date.now(), 2, 'month'), 10) }); + + await engine.syncTables(); + + rows = await engine.search('created:month-0'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('created:month-1'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('created:month-2'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + })); + + it('should support filtering by created with smart value: year', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'I made this', body: 'this year', user_created_time: parseInt(time.goBackInTime(Date.now(), 0, 'year'), 10) }); + const n2 = await Note.save({ title: 'I made this', body: 'the year before', user_created_time: parseInt(time.goBackInTime(Date.now(), 1, 'year'), 10) }); + const n3 = await Note.save({ title: 'I made this', body: 'before before year', user_created_time: parseInt(time.goBackInTime(Date.now(), 2, 'year'), 10) }); + + await engine.syncTables(); + + rows = await engine.search('created:year-0'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('created:year-1'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('created:year-2'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + })); + + it('should support filtering by updated date', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'I updated this on', body: 'May 20 2020', updated_time: Date.parse('2020-05-20'), user_updated_time: Date.parse('2020-05-20') }, { autoTimestamp: false }); + const n2 = await Note.save({ title: 'I updated this on', body: 'May 19 2020', updated_time: Date.parse('2020-05-19'), user_updated_time: Date.parse('2020-05-19') }, { autoTimestamp: false }); + + await engine.syncTables(); + + rows = await engine.search('updated:20200520'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('updated:20200519'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + })); + + it('should support filtering by updated with smart value: day', asyncTest(async () => { + let rows; + const today = parseInt(time.goBackInTime(Date.now(), 0, 'day'), 10); + const yesterday = parseInt(time.goBackInTime(Date.now(), 1, 'day'), 10); + const dayBeforeYesterday = parseInt(time.goBackInTime(Date.now(), 2, 'day'), 10); + const n1 = await Note.save({ title: 'I made this', body: 'today', updated_time: today, user_updated_time: today }, { autoTimestamp: false }); + const n11 = await Note.save({ title: 'I also made this', body: 'today', updated_time: today, user_updated_time: today }, { autoTimestamp: false }); + + const n2 = await Note.save({ title: 'I made this', body: 'yesterday', updated_time: yesterday, user_updated_time: yesterday }, { autoTimestamp: false }); + const n3 = await Note.save({ title: 'I made this', body: 'day before yesterday', updated_time: dayBeforeYesterday ,user_updated_time: dayBeforeYesterday }, { autoTimestamp: false }); + + await engine.syncTables(); + + rows = await engine.search('updated:day-0'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n11.id); + + rows = await engine.search('updated:day-1'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n11.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('updated:day-2'); + expect(rows.length).toBe(4); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n11.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + })); + + it('should support filtering by type todo', asyncTest(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 }); + const t3 = await Note.save({ title: 'This is NOT a ', body: 'todo' }); + + await engine.syncTables(); + + rows = await engine.search('type:todo'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(t1.id); + expect(ids(rows)).toContain(t2.id); + + rows = await engine.search('any:1 type:todo'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(t1.id); + expect(ids(rows)).toContain(t2.id); + + rows = await engine.search('iscompleted:1'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(t2.id); + + rows = await engine.search('iscompleted:0'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(t1.id); + })); + + it('should support filtering by type note', asyncTest(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 }); + const t3 = await Note.save({ title: 'This is NOT a ', body: 'todo' }); + + await engine.syncTables(); + + rows = await engine.search('type:note'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(t3.id); + })); + + it('should support filtering by latitude, longitude, altitude', asyncTest(async () => { + let rows; + const n1 = await Note.save({ title: 'I made this', body: 'this week', latitude: 12.97, longitude: 88.88, altitude: 69.96 }); + const n2 = await Note.save({ title: 'I made this', body: 'the week before', latitude: 42.11, longitude: 77.77, altitude: 42.00 }); + const n3 = await Note.save({ title: 'I made this', body: 'before before week', latitude: 82.01, longitude: 66.66, altitude: 13.13 }); + + await engine.syncTables(); + + rows = await engine.search('latitude:13.5'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('-latitude:40'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('latitude:13 -latitude:80'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('altitude:13.5'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('-altitude:80.12'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('longitude:70 -longitude:80'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('latitude:20 longitude:50 altitude:40'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('any:1 latitude:20 longitude:50 altitude:40'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + })); + + it('should support filtering by resource MIME type', asyncTest(async () => { + let rows; + const service = new ResourceService(); + // console.log(testImagePath) + const folder1 = await Folder.save({ title: 'folder1' }); + let n1 = await Note.save({ title: 'I have a picture', body: 'Im awesome', parent_id: folder1.id }); + const n2 = await Note.save({ title: 'Boring note 1', body: 'I just have text', parent_id: folder1.id }); + const n3 = await Note.save({ title: 'Boring note 2', body: 'me too', parent_id: folder1.id }); + let n4 = await Note.save({ title: 'A picture?', body: 'pfff, I have a pdf', parent_id: folder1.id }); + await engine.syncTables(); + + // let note1 = await Note.save({ title: 'ma note', parent_id: folder1.id }); + n1 = await shim.attachFileToNote(n1, `${__dirname}/../tests/support/photo.jpg`); + // const resource1 = (await Resource.all())[0]; + + n4 = await shim.attachFileToNote(n4, `${__dirname}/../tests/support/welcome.pdf`); + + await service.indexNoteResources(); + + rows = await engine.search('resource:image/jpeg'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('resource:image/*'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('resource:application/pdf'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n4.id); + + rows = await engine.search('-resource:image/jpeg'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + expect(ids(rows)).toContain(n4.id); + + rows = await engine.search('any:1 resource:application/pdf resource:image/jpeg'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n4.id); + })); + + it('should ignore dashes in a word', asyncTest(async () => { + const n0 = await Note.save({ title: 'doesnotwork' }); + const n1 = await Note.save({ title: 'does not work' }); + const n2 = await Note.save({ title: 'does-not-work' }); + const n3 = await Note.save({ title: 'does_not_work' }); + + await engine.syncTables(); + + let rows = await engine.search('does-not-work'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('does not work'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('"does not work"'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('title:does-not-work'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('doesnotwork'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n0.id); + + })); + + it('should support filtering by sourceurl', asyncTest(async () => { + const n0 = await Note.save({ title: 'n0', source_url: 'https://discourse.joplinapp.org' }); + const n1 = await Note.save({ title: 'n1', source_url: 'https://google.com' }); + const n2 = await Note.save({ title: 'n2', source_url: 'https://reddit.com' }); + const n3 = await Note.save({ title: 'n3', source_url: 'https://joplinapp.org' }); + + await engine.syncTables(); + + let rows = await engine.search('sourceurl:https://joplinapp.org'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('sourceurl:https://google.com'); + expect(rows.length).toBe(1); + expect(ids(rows)).toContain(n1.id); + + rows = await engine.search('any:1 sourceurl:https://google.com sourceurl:https://reddit.com'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n1.id); + expect(ids(rows)).toContain(n2.id); + + rows = await engine.search('-sourceurl:https://google.com'); + expect(rows.length).toBe(3); + expect(ids(rows)).toContain(n0.id); + expect(ids(rows)).toContain(n2.id); + expect(ids(rows)).toContain(n3.id); + + rows = await engine.search('sourceurl:*joplinapp.org'); + expect(rows.length).toBe(2); + expect(ids(rows)).toContain(n0.id); + expect(ids(rows)).toContain(n3.id); + + })); + +}); diff --git a/CliClient/tests/support/welcome.pdf b/CliClient/tests/support/welcome.pdf new file mode 100644 index 0000000000000000000000000000000000000000..cf34d95226cf4f2dbc2fd55a1ad00a9438b49e70 GIT binary patch literal 6736 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 Pgn+|XPMkP>PWOKRT0nVW literal 0 HcmV?d00001 diff --git a/CliClient/tests/timeUtils.js b/CliClient/tests/timeUtils.js new file mode 100644 index 000000000..de7987bb9 --- /dev/null +++ b/CliClient/tests/timeUtils.js @@ -0,0 +1,66 @@ +/* eslint-disable no-unused-vars */ + +require('app-module-path').addPath(__dirname); + +const { time } = require('lib/time-utils.js'); +const { asyncTest, fileContentEqual, setupDatabase, setupDatabaseAndSynchronizer, db, synchronizer, fileApi, sleep, clearDatabase, switchClient, syncTargetId, objectsEqual, checkThrowAsync } = require('test-utils.js'); +const timeUtils = require('../../ReactNativeClient/lib/time-utils'); + +process.on('unhandledRejection', (reason, p) => { + console.log('Unhandled Rejection at: Promise', p, 'reason:', reason); +}); + +describe('timeUtils', function() { + + beforeEach(async (done) => { + done(); + }); + + it('should go back in time', asyncTest(async () => { + let startDate = new Date('3 Aug 2020'); + let endDate = new Date('2 Aug 2020'); + + expect(time.goBackInTime(startDate, 1, 'day')).toBe(endDate.getTime().toString()); + + // We're always subtracting time from the beginning of the current period. + startDate = new Date('3 Aug 2020 07:30:20'); + expect(time.goBackInTime(startDate, 1, 'day')).toBe(endDate.getTime().toString()); + + + startDate = new Date('11 Aug 2020'); + endDate = new Date('9 Aug 2020'); // week start; + expect(time.goBackInTime(startDate, 0, 'week')).toBe(endDate.getTime().toString()); + + startDate = new Date('02 Feb 2020'); + endDate = new Date('01 Jan 2020'); + expect(time.goBackInTime(startDate, 1, 'month')).toBe(endDate.getTime().toString()); + + startDate = new Date('19 September 2020'); + endDate = new Date('01 Jan 1997'); + expect(time.goBackInTime(startDate, 23, 'year')).toBe(endDate.getTime().toString()); + })); + + it('should go forward in time', asyncTest(async () => { + let startDate = new Date('2 Aug 2020'); + let endDate = new Date('3 Aug 2020'); + + expect(time.goForwardInTime(startDate, 1, 'day')).toBe(endDate.getTime().toString()); + + startDate = new Date('2 Aug 2020 07:30:20'); + expect(time.goForwardInTime(startDate, 1, 'day')).toBe(endDate.getTime().toString()); + + + startDate = new Date('9 Aug 2020'); + endDate = new Date('9 Aug 2020'); // week start; + expect(time.goForwardInTime(startDate, 0, 'week')).toBe(endDate.getTime().toString()); + + startDate = new Date('02 Jan 2020'); + endDate = new Date('01 Feb 2020'); + expect(time.goForwardInTime(startDate, 1, 'month')).toBe(endDate.getTime().toString()); + + startDate = new Date('19 September 1997'); + endDate = new Date('01 Jan 2020'); + expect(time.goForwardInTime(startDate, 23, 'year')).toBe(endDate.getTime().toString()); + })); + +}); diff --git a/ElectronClient/gui/NoteEditor/utils/useSearchMarkers.ts b/ElectronClient/gui/NoteEditor/utils/useSearchMarkers.ts index a168bb93c..a69d6bf81 100644 --- a/ElectronClient/gui/NoteEditor/utils/useSearchMarkers.ts +++ b/ElectronClient/gui/NoteEditor/utils/useSearchMarkers.ts @@ -1,7 +1,7 @@ import { useMemo } from 'react'; const BaseModel = require('lib/BaseModel.js'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); interface SearchMarkersOptions { searchTimestamp: number, diff --git a/ElectronClient/gui/NoteList/NoteList.jsx b/ElectronClient/gui/NoteList/NoteList.jsx index d1ee63779..1229492ff 100644 --- a/ElectronClient/gui/NoteList/NoteList.jsx +++ b/ElectronClient/gui/NoteList/NoteList.jsx @@ -7,7 +7,7 @@ const BaseModel = require('lib/BaseModel'); const { _ } = require('lib/locale.js'); const { bridge } = require('electron').remote.require('./bridge'); const eventManager = require('lib/eventManager'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const Note = require('lib/models/Note'); const Setting = require('lib/models/Setting'); const NoteListUtils = require('../utils/NoteListUtils'); diff --git a/ElectronClient/plugins/GotoAnything.jsx b/ElectronClient/plugins/GotoAnything.jsx index 6206d99cf..bfadde6dc 100644 --- a/ElectronClient/plugins/GotoAnything.jsx +++ b/ElectronClient/plugins/GotoAnything.jsx @@ -2,8 +2,8 @@ const React = require('react'); const { connect } = require('react-redux'); const { _ } = require('lib/locale.js'); const { themeStyle } = require('lib/theme'); -const SearchEngine = require('lib/services/SearchEngine'); const CommandService = require('lib/services/CommandService').default; +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const BaseModel = require('lib/BaseModel'); const Tag = require('lib/models/Tag'); const Folder = require('lib/models/Folder'); diff --git a/ReactNativeClient/lib/BaseApplication.js b/ReactNativeClient/lib/BaseApplication.js index 6dfc377be..7d18be6ca 100644 --- a/ReactNativeClient/lib/BaseApplication.js +++ b/ReactNativeClient/lib/BaseApplication.js @@ -32,12 +32,12 @@ const SyncTargetDropbox = require('lib/SyncTargetDropbox.js'); const SyncTargetAmazonS3 = require('lib/SyncTargetAmazonS3.js'); const EncryptionService = require('lib/services/EncryptionService'); const ResourceFetcher = require('lib/services/ResourceFetcher'); -const SearchEngineUtils = require('lib/services/SearchEngineUtils'); +const SearchEngineUtils = require('lib/services/searchengine/SearchEngineUtils'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const RevisionService = require('lib/services/RevisionService'); const ResourceService = require('lib/services/RevisionService'); const DecryptionWorker = require('lib/services/DecryptionWorker'); const BaseService = require('lib/services/BaseService'); -const SearchEngine = require('lib/services/SearchEngine'); const { loadKeychainServiceAndSettings } = require('lib/services/SettingUtils'); const KeychainServiceDriver = require('lib/services/keychain/KeychainServiceDriver.node').default; const KvStore = require('lib/services/KvStore'); diff --git a/ReactNativeClient/lib/components/screens/config.js b/ReactNativeClient/lib/components/screens/config.js index 40f396cfc..38825bae0 100644 --- a/ReactNativeClient/lib/components/screens/config.js +++ b/ReactNativeClient/lib/components/screens/config.js @@ -16,7 +16,7 @@ const VersionInfo = require('react-native-version-info').default; const { ReportService } = require('lib/services/report.js'); const { time } = require('lib/time-utils'); const { shim } = require('lib/shim'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const RNFS = require('react-native-fs'); const checkPermissions = require('lib/checkPermissions.js').default; diff --git a/ReactNativeClient/lib/components/screens/note.js b/ReactNativeClient/lib/components/screens/note.js index e84019428..e271da597 100644 --- a/ReactNativeClient/lib/components/screens/note.js +++ b/ReactNativeClient/lib/components/screens/note.js @@ -40,7 +40,7 @@ const ImagePicker = require('react-native-image-picker'); const { SelectDateTimeDialog } = require('lib/components/select-date-time-dialog.js'); const ShareExtension = require('lib/ShareExtension.js').default; const CameraView = require('lib/components/CameraView'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const urlUtils = require('lib/urlUtils'); class NoteScreenComponent extends BaseScreenComponent { diff --git a/ReactNativeClient/lib/components/screens/search.js b/ReactNativeClient/lib/components/screens/search.js index 789df7f66..e05f75477 100644 --- a/ReactNativeClient/lib/components/screens/search.js +++ b/ReactNativeClient/lib/components/screens/search.js @@ -9,8 +9,8 @@ const Note = require('lib/models/Note.js'); const { NoteItem } = require('lib/components/note-item.js'); const { BaseScreenComponent } = require('lib/components/base-screen.js'); const { themeStyle } = require('lib/components/global-style.js'); -const SearchEngineUtils = require('lib/services/SearchEngineUtils'); const DialogBox = require('react-native-dialogbox').default; +const SearchEngineUtils = require('lib/services/searchengine/SearchEngineUtils'); Icon.loadFont(); diff --git a/ReactNativeClient/lib/joplin-database.js b/ReactNativeClient/lib/joplin-database.js index 87cbc1b8b..ab0a2897a 100644 --- a/ReactNativeClient/lib/joplin-database.js +++ b/ReactNativeClient/lib/joplin-database.js @@ -326,7 +326,7 @@ class JoplinDatabase extends Database { // must be set in the synchronizer too. // Note: v16 and v17 don't do anything. They were used to debug an issue. - const existingDatabaseVersions = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32]; + const existingDatabaseVersions = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33]; let currentVersionIndex = existingDatabaseVersions.indexOf(fromVersion); @@ -757,13 +757,96 @@ class JoplinDatabase extends Database { GROUP BY tags.id`); } + if (targetVersion == 33) { + queries.push('DROP TRIGGER notes_fts_before_update'); + queries.push('DROP TRIGGER notes_fts_before_delete'); + queries.push('DROP TRIGGER notes_after_update'); + queries.push('DROP TRIGGER notes_after_insert'); + + queries.push('DROP INDEX notes_normalized_id'); + queries.push('DROP TABLE notes_normalized'); + queries.push('DROP TABLE notes_fts'); + + + const notesNormalized = ` + CREATE TABLE notes_normalized ( + id TEXT NOT NULL, + title TEXT NOT NULL DEFAULT "", + body TEXT NOT NULL DEFAULT "", + user_created_time INT NOT NULL DEFAULT 0, + user_updated_time INT NOT NULL DEFAULT 0, + is_todo INT NOT NULL DEFAULT 0, + todo_completed INT NOT NULL DEFAULT 0, + parent_id TEXT NOT NULL DEFAULT "", + latitude NUMERIC NOT NULL DEFAULT 0, + longitude NUMERIC NOT NULL DEFAULT 0, + altitude NUMERIC NOT NULL DEFAULT 0, + source_url TEXT NOT NULL DEFAULT "" + ); + `; + + queries.push(this.sqlStringToLines(notesNormalized)[0]); + + queries.push('CREATE INDEX notes_normalized_id ON notes_normalized (id)'); + + queries.push('CREATE INDEX notes_normalized_user_created_time ON notes_normalized (user_created_time)'); + queries.push('CREATE INDEX notes_normalized_user_updated_time ON notes_normalized (user_updated_time)'); + queries.push('CREATE INDEX notes_normalized_is_todo ON notes_normalized (is_todo)'); + queries.push('CREATE INDEX notes_normalized_todo_completed ON notes_normalized (todo_completed)'); + queries.push('CREATE INDEX notes_normalized_parent_id ON notes_normalized (parent_id)'); + queries.push('CREATE INDEX notes_normalized_latitude ON notes_normalized (latitude)'); + queries.push('CREATE INDEX notes_normalized_longitude ON notes_normalized (longitude)'); + queries.push('CREATE INDEX notes_normalized_altitude ON notes_normalized (altitude)'); + queries.push('CREATE INDEX notes_normalized_source_url ON notes_normalized (source_url)'); + + const tableFields = 'id, title, body, user_created_time, user_updated_time, is_todo, todo_completed, parent_id, latitude, longitude, altitude, source_url'; + + + const newVirtualTableSql = ` + CREATE VIRTUAL TABLE notes_fts USING fts4( + content="notes_normalized", + notindexed="id", + notindexed="user_created_time", + notindexed="user_updated_time", + notindexed="is_todo", + notindexed="todo_completed", + notindexed="parent_id", + notindexed="latitude", + notindexed="longitude", + notindexed="altitude", + notindexed="source_url", + ${tableFields} + );` + ; + + queries.push(this.sqlStringToLines(newVirtualTableSql)[0]); + + queries.push(` + CREATE TRIGGER notes_fts_before_update BEFORE UPDATE ON notes_normalized BEGIN + DELETE FROM notes_fts WHERE docid=old.rowid; + END;`); + queries.push(` + CREATE TRIGGER notes_fts_before_delete BEFORE DELETE ON notes_normalized BEGIN + DELETE FROM notes_fts WHERE docid=old.rowid; + END;`); + queries.push(` + CREATE TRIGGER notes_after_update AFTER UPDATE ON notes_normalized BEGIN + INSERT INTO notes_fts(docid, ${tableFields}) SELECT rowid, ${tableFields} FROM notes_normalized WHERE new.rowid = notes_normalized.rowid; + END;`); + queries.push(` + CREATE TRIGGER notes_after_insert AFTER INSERT ON notes_normalized BEGIN + INSERT INTO notes_fts(docid, ${tableFields}) SELECT rowid, ${tableFields} FROM notes_normalized WHERE new.rowid = notes_normalized.rowid; + END;`); + queries.push(this.addMigrationFile(33)); + } + queries.push({ sql: 'UPDATE version SET version = ?', params: [targetVersion] }); try { await this.transactionExecBatch(queries); } catch (error) { - if (targetVersion === 15 || targetVersion === 18) { - this.logger().warn('Could not upgrade to database v15 or v18 - FTS feature will not be used', error); + if (targetVersion === 15 || targetVersion === 18 || targetVersion === 33) { + this.logger().warn('Could not upgrade to database v15 or v18 or v33- FTS feature will not be used', error); } else { throw error; } diff --git a/ReactNativeClient/lib/migrations/33.js b/ReactNativeClient/lib/migrations/33.js new file mode 100644 index 000000000..f9164e5b0 --- /dev/null +++ b/ReactNativeClient/lib/migrations/33.js @@ -0,0 +1,9 @@ +const SearchEngine = require('lib/services/searchengine/SearchEngine'); + +const script = {}; + +script.exec = async function() { + await SearchEngine.instance().rebuildIndex(); +}; + +module.exports = script; diff --git a/ReactNativeClient/lib/models/Migration.js b/ReactNativeClient/lib/models/Migration.js index 8e7964c45..1843ac40d 100644 --- a/ReactNativeClient/lib/models/Migration.js +++ b/ReactNativeClient/lib/models/Migration.js @@ -3,6 +3,7 @@ const BaseModel = require('lib/BaseModel.js'); const migrationScripts = { 20: require('lib/migrations/20.js'), 27: require('lib/migrations/27.js'), + 33: require('lib/migrations/33.js'), }; class Migration extends BaseModel { diff --git a/ReactNativeClient/lib/services/ResourceService.js b/ReactNativeClient/lib/services/ResourceService.js index 302b36fc8..772c63b8d 100644 --- a/ReactNativeClient/lib/services/ResourceService.js +++ b/ReactNativeClient/lib/services/ResourceService.js @@ -4,7 +4,7 @@ const Note = require('lib/models/Note'); const Resource = require('lib/models/Resource'); const BaseModel = require('lib/BaseModel'); const BaseService = require('lib/services/BaseService'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const Setting = require('lib/models/Setting'); const { shim } = require('lib/shim'); const ItemChangeUtils = require('lib/services/ItemChangeUtils'); diff --git a/ReactNativeClient/lib/services/rest/Api.js b/ReactNativeClient/lib/services/rest/Api.js index ae8ff6674..717cfb437 100644 --- a/ReactNativeClient/lib/services/rest/Api.js +++ b/ReactNativeClient/lib/services/rest/Api.js @@ -19,7 +19,7 @@ const ArrayUtils = require('lib/ArrayUtils.js'); const { netUtils } = require('lib/net-utils'); const { fileExtension, safeFileExtension, safeFilename, filename } = require('lib/path-utils'); const ApiResponse = require('lib/services/rest/ApiResponse'); -const SearchEngineUtils = require('lib/services/SearchEngineUtils'); +const SearchEngineUtils = require('lib/services/searchengine/SearchEngineUtils'); const { FoldersScreenUtils } = require('lib/folders-screen-utils.js'); const uri2path = require('file-uri-to-path'); const { MarkupToHtml } = require('lib/joplin-renderer'); diff --git a/ReactNativeClient/lib/services/SearchEngine.js b/ReactNativeClient/lib/services/searchengine/SearchEngine.js similarity index 83% rename from ReactNativeClient/lib/services/SearchEngine.js rename to ReactNativeClient/lib/services/searchengine/SearchEngine.js index 3b964e88b..9d3a41b3c 100644 --- a/ReactNativeClient/lib/services/SearchEngine.js +++ b/ReactNativeClient/lib/services/searchengine/SearchEngine.js @@ -7,8 +7,11 @@ const ItemChangeUtils = require('lib/services/ItemChangeUtils'); const { pregQuote, scriptType } = require('lib/string-utils.js'); const removeDiacritics = require('diacritics').remove; const { sprintf } = require('sprintf-js'); +const filterParser = require('./filterParser').default; +const queryBuilder = require('./queryBuilder').default; class SearchEngine { + constructor() { this.dispatch = () => {}; this.logger_ = new Logger(); @@ -62,13 +65,20 @@ class SearchEngine { while (noteIds.length) { const currentIds = noteIds.splice(0, 100); - const notes = await Note.modelSelectAll(`SELECT id, title, body FROM notes WHERE id IN ("${currentIds.join('","')}") AND is_conflict = 0 AND encryption_applied = 0`); + const notes = await Note.modelSelectAll(` + SELECT ${SearchEngine.relevantFields} + FROM notes + WHERE id IN ("${currentIds.join('","')}") AND is_conflict = 0 AND encryption_applied = 0`); const queries = []; for (let i = 0; i < notes.length; i++) { const note = notes[i]; const n = this.normalizeNote_(note); - queries.push({ sql: 'INSERT INTO notes_normalized(id, title, body) VALUES (?, ?, ?)', params: [n.id, n.title, n.body] }); + queries.push({ sql: ` + INSERT INTO notes_normalized(${SearchEngine.relevantFields}) + VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`, + params: [n.id, n.title, n.body, n.user_created_time, n.user_updated_time, n.is_todo, n.todo_completed, n.parent_id, n.latitude, n.longitude, n.altitude, n.source_url] } + ); } await this.db().transactionExecBatch(queries); @@ -138,7 +148,11 @@ class SearchEngine { if (!changes.length) break; const noteIds = changes.map(a => a.item_id); - const notes = await Note.modelSelectAll(`SELECT id, title, body FROM notes WHERE id IN ("${noteIds.join('","')}") AND is_conflict = 0 AND encryption_applied = 0`); + const notes = await Note.modelSelectAll(` + SELECT ${SearchEngine.relevantFields} + FROM notes WHERE id IN ("${noteIds.join('","')}") AND is_conflict = 0 AND encryption_applied = 0` + ); + const queries = []; for (let i = 0; i < changes.length; i++) { @@ -149,7 +163,10 @@ class SearchEngine { const note = this.noteById_(notes, change.item_id); if (note) { const n = this.normalizeNote_(note); - queries.push({ sql: 'INSERT INTO notes_normalized(id, title, body) VALUES (?, ?, ?)', params: [change.item_id, n.title, n.body] }); + queries.push({ sql: ` + INSERT INTO notes_normalized(${SearchEngine.relevantFields}) + VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`, + params: [change.item_id, n.title, n.body, n.user_created_time, n.user_updated_time, n.is_todo, n.todo_completed, n.parent_id, n.latitude, n.longitude, n.altitude, n.source_url] }); report.inserted++; } } else if (change.type === ItemChange.TYPE_DELETE) { @@ -295,44 +312,20 @@ class SearchEngine { } parseQuery(query) { - const terms = { _: [] }; + const trimQuotes = (str) => str.startsWith('"') ? str.substr(1, str.length - 2) : str; - let inQuote = false; - let currentCol = '_'; - let currentTerm = ''; - for (let i = 0; i < query.length; i++) { - const c = query[i]; - - if (c === '"') { - if (inQuote) { - terms[currentCol].push(currentTerm); - currentTerm = ''; - inQuote = false; - } else { - inQuote = true; - } - continue; - } - - if (c === ' ' && !inQuote) { - if (!currentTerm) continue; - terms[currentCol].push(currentTerm); - currentCol = '_'; - currentTerm = ''; - continue; - } - - if (c === ':' && !inQuote) { - currentCol = currentTerm; - if (!terms[currentCol]) terms[currentCol] = []; - currentTerm = ''; - continue; - } - - currentTerm += c; + let allTerms = []; + try { + allTerms = filterParser(query); + } catch (error) { + console.warn(error); } - if (currentTerm) terms[currentCol].push(currentTerm); + const textTerms = allTerms.filter(x => x.name === 'text').map(x => trimQuotes(x.value)); + const titleTerms = allTerms.filter(x => x.name === 'title').map(x => trimQuotes(x.value)); + const bodyTerms = allTerms.filter(x => x.name === 'body').map(x => trimQuotes(x.value)); + + const terms = { _: textTerms, 'title': titleTerms, 'body': bodyTerms }; // Filter terms: // - Convert wildcards to regex @@ -373,7 +366,8 @@ class SearchEngine { return { termCount: termCount, keys: keys, - terms: terms, + terms: terms, // text terms + allTerms: allTerms, }; } @@ -432,54 +426,38 @@ class SearchEngine { return SearchEngine.SEARCH_TYPE_FTS; } - async search(query, options = null) { + async search(searchString, options = null) { options = Object.assign({}, { searchType: SearchEngine.SEARCH_TYPE_AUTO, }, options); - query = this.normalizeText_(query); + searchString = this.normalizeText_(searchString); - const searchType = this.determineSearchType_(query, options.searchType); - const parsedQuery = this.parseQuery(query); + const searchType = this.determineSearchType_(searchString, options.searchType); if (searchType === SearchEngine.SEARCH_TYPE_BASIC) { // Non-alphabetical languages aren't support by SQLite FTS (except with extensions which are not available in all platforms) - const rows = await this.basicSearch(query); + const rows = await this.basicSearch(searchString); + const parsedQuery = this.parseQuery(searchString); this.processResults_(rows, parsedQuery, true); return rows; - } else { // SEARCH_TYPE_FTS + } else { + // SEARCH_TYPE_FTS // FTS will ignore all special characters, like "-" in the index. So if // we search for "this-phrase" it won't find it because it will only // see "this phrase" in the index. Because of this, we remove the dashes // when searching. // https://github.com/laurent22/joplin/issues/1075#issuecomment-459258856 - query = query.replace(/-/g, ' '); - // Note that when the search engine index is somehow corrupted, it might contain - // references to notes that don't exist. Not clear how it can happen, but anyway - // handle it here by checking if `user_updated_time` IS NOT NULL. Was causing this - // issue: https://discourse.joplinapp.org/t/how-to-recover-corrupted-database/9367 - const sql = ` - SELECT - notes_fts.id, - notes_fts.title AS normalized_title, - offsets(notes_fts) AS offsets, - notes.title, - notes.user_updated_time, - notes.is_todo, - notes.todo_completed, - notes.parent_id - FROM notes_fts - LEFT JOIN notes ON notes_fts.id = notes.id - WHERE notes_fts MATCH ? - AND notes.user_updated_time IS NOT NULL - `; + const parsedQuery = this.parseQuery(searchString); + try { - const rows = await this.db().selectAll(sql, [query]); + const { query, params } = queryBuilder(parsedQuery.allTerms); + const rows = await this.db().selectAll(query, params); this.processResults_(rows, parsedQuery); return rows; } catch (error) { - this.logger().warn(`Cannot execute MATCH query: ${query}: ${error.message}`); + this.logger().warn(`Cannot execute MATCH query: ${searchString}: ${error.message}`); return []; } } @@ -504,6 +482,8 @@ class SearchEngine { } } +SearchEngine.relevantFields = 'id, title, body, user_created_time, user_updated_time, is_todo, todo_completed, parent_id, latitude, longitude, altitude, source_url'; + SearchEngine.instance_ = null; SearchEngine.SEARCH_TYPE_AUTO = 'auto'; diff --git a/ReactNativeClient/lib/services/SearchEngineUtils.js b/ReactNativeClient/lib/services/searchengine/SearchEngineUtils.js similarity index 87% rename from ReactNativeClient/lib/services/SearchEngineUtils.js rename to ReactNativeClient/lib/services/searchengine/SearchEngineUtils.js index 98cd2efaf..1384a7939 100644 --- a/ReactNativeClient/lib/services/SearchEngineUtils.js +++ b/ReactNativeClient/lib/services/searchengine/SearchEngineUtils.js @@ -1,4 +1,4 @@ -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const Note = require('lib/models/Note'); class SearchEngineUtils { @@ -42,7 +42,13 @@ class SearchEngineUtils { if (idWasAutoAdded) delete sortedNotes[idx].id; } - return sortedNotes; + if (noteIds.length !== notes.length) { + // remove null objects + return sortedNotes.filter(n => n); + } else { + return sortedNotes; + } + } } diff --git a/ReactNativeClient/lib/services/searchengine/filterParser.ts b/ReactNativeClient/lib/services/searchengine/filterParser.ts new file mode 100644 index 000000000..6bbbfac7d --- /dev/null +++ b/ReactNativeClient/lib/services/searchengine/filterParser.ts @@ -0,0 +1,133 @@ + +interface Term { + name: string + value: string + negated: boolean +} + +const makeTerm = (name: string, value: string): Term => { + if (name.startsWith('-')) { return { name: name.slice(1), value: value, negated: true }; } + return { name: name, value: value, negated: false }; +}; + + +const quote = (s : string) => { + const quoted = (s: string) => s.startsWith('"') && s.endsWith('"'); + + if (!quoted(s)) { + return `"${s}"`; + } + return s; +}; + + +const getTerms = (query: string) : Term[] => { + const terms: Term[] = []; + let inQuote = false; + let inTerm = false; + let currentCol = '_'; + let currentTerm = ''; + for (let i = 0; i < query.length; i++) { + const c = query[i]; + + if (c === '"') { + currentTerm += c; // keep the quotes + if (inQuote) { + terms.push(makeTerm(currentCol, currentTerm)); + currentTerm = ''; + inQuote = false; + } else { + inQuote = true; + } + continue; + } + + if (c === ' ' && !inQuote) { + inTerm = false; + if (!currentTerm) continue; + terms.push(makeTerm(currentCol, currentTerm)); + currentCol = '_'; + currentTerm = ''; + continue; + } + + if (c === ':' && !inQuote && !inTerm) { + currentCol = currentTerm; + currentTerm = ''; + inTerm = true; // to ignore any other ':' before a space eg.'sourceurl:https://www.google.com' + continue; + } + + currentTerm += c; + } + if (currentTerm) terms.push(makeTerm(currentCol, currentTerm)); + return terms; +}; + +const parseQuery = (query: string): Term[] => { + const validFilters = new Set(['any', 'title', 'body', 'tag', + 'notebook', 'created', 'updated', 'type', + 'iscompleted', 'latitude', 'longitude', + 'altitude', 'resource', 'sourceurl']); + + const terms = getTerms(query); + // console.log(terms); + + const result: Term[] = []; + for (let i = 0; i < terms.length; i++) { + const { name, value, negated } = terms[i]; + + if (name !== '_') { + if (!validFilters.has(name)) { + throw new Error(`Invalid filter: ${name}`); + } + + if (name === 'tag' || name === 'notebook' || name === 'resource' || name === 'sourceurl') { + result.push({ name, value: value.replace(/[*]/g, '%'), negated }); // for wildcard search + } else if (name === 'title' || name === 'body') { + // Trim quotes since we don't support phrase query here + // eg. Split title:"hello world" to title:hello title:world + const values = trimQuotes(value).split(/[\s-_]+/); + values.forEach(value => { + result.push({ name, value, negated }); + }); + } else { + result.push({ name, value, negated }); + } + } else { + // Every word is quoted if not already. + // By quoting the word, FTS match query will take care of removing dashes and other word seperators. + if (value.startsWith('-')) { + result.push({ name: 'text', value: quote(value.slice(1)) , negated: true }); + } else { + result.push({ name: 'text', value: quote(value), negated: false }); + } + } + } + + // validation + let incorrect = result.filter(term => term.name === 'type' || term.name === 'iscompleted' || term.name === 'notebook') + .find(x => x.negated); + if (incorrect) throw new Error(`${incorrect.name} can't be negated`); + + incorrect = result.filter(term => term.name === 'type') + .find(x => (x.value !== 'note' && x.value !== 'todo')); + if (incorrect) throw new Error('The value of filter "type" must be "note" or "todo"'); + + incorrect = result.filter(term => term.name === 'iscompleted') + .find(x => (x.value !== '1' && x.value !== '0')); + if (incorrect) throw new Error('The value of filter "iscompleted" must be "1" or "0"'); + + + return result; +}; + +const trimQuotes = (str: string): string => str.startsWith('"') ? str.substr(1, str.length - 2) : str; + +export default function filterParser(searchString: string) { + searchString = searchString.trim(); + + const result = parseQuery(searchString); + + return result; +} diff --git a/ReactNativeClient/lib/services/searchengine/queryBuilder.ts b/ReactNativeClient/lib/services/searchengine/queryBuilder.ts new file mode 100644 index 000000000..3c4d0f7d2 --- /dev/null +++ b/ReactNativeClient/lib/services/searchengine/queryBuilder.ts @@ -0,0 +1,428 @@ +const { time } = require('lib/time-utils.js'); + +interface Term { + name: string + value: string + negated: boolean +} + +enum Relation { + OR = 'OR', + AND = 'AND', +} + +enum Operation { + UNION = 'UNION', + INTERSECT = 'INTERSECT' +} + +enum Requirement { + EXCLUSION = 'EXCLUSION', + INCLUSION = 'INCLUSION' +} + +const notebookFilter = (terms: Term[], conditions: string[], params: string[], withs: string[]) => { + const notebooks = terms.filter(x => x.name === 'notebook' && !x.negated).map(x => x.value); + if (notebooks.length === 0) return; + + const likes = []; + for (let i = 0; i < notebooks.length; i++) { + likes.push('folders.title LIKE ?'); + } + const relevantFolders = likes.join(' OR '); + + const withInNotebook = ` + notebooks_in_scope(id) + AS ( + SELECT folders.id + FROM folders + WHERE id + IN ( + SELECT id + FROM folders + WHERE ${relevantFolders} + ) + UNION ALL + SELECT folders.id + FROM folders + JOIN notebooks_in_scope + ON folders.parent_id=notebooks_in_scope.id + )`; + const where = ` + AND ROWID IN ( + SELECT notes_normalized.ROWID + FROM notebooks_in_scope + JOIN notes_normalized + ON notebooks_in_scope.id=notes_normalized.parent_id + )`; + + withs.push(withInNotebook); + params.push(...notebooks); + conditions.push(where); +}; + +const getOperator = (requirement: Requirement, relation: Relation): Operation => { + if (relation === 'AND' && requirement === 'INCLUSION') { return Operation.INTERSECT; } else { return Operation.UNION; } +}; + +const filterByTableName = ( + terms: Term[], + conditions: string[], + params: string[], + relation: Relation, + noteIDs: string, + requirement: Requirement, + withs: string[], + tableName: string +) => { + const operator: Operation = getOperator(requirement, relation); + + const values = terms.map(x => x.value); + + let withCondition = null; + + if (relation === Relation.OR && requirement === Requirement.EXCLUSION) { + // with_${requirement}_${tableName} is added to the names to make them unique + withs.push(` + all_notes_with_${requirement}_${tableName} + AS ( + SELECT DISTINCT note_${tableName}.note_id AS id FROM note_${tableName} + )`); + + const notesWithoutExcludedField = ` + SELECT * FROM ( + SELECT * + FROM all_notes_with_${requirement}_${tableName} + EXCEPT ${noteIDs} + )`; + + const requiredNotes = []; + for (let i = 0; i < values.length; i++) { + requiredNotes.push(notesWithoutExcludedField); + } + const requiredNotesQuery = requiredNotes.join(' UNION '); + + // We need notes without atleast one excluded (tag/resource) + withCondition = ` + notes_with_${requirement}_${tableName} + AS ( + ${requiredNotesQuery} + )`; + + } else { + const requiredNotes = []; + for (let i = 0; i < values.length; i++) { + requiredNotes.push(noteIDs); + } + const requiredNotesQuery = requiredNotes.join(` ${operator} `); + + + // Notes with any/all values depending upon relation and requirement + withCondition = ` + notes_with_${requirement}_${tableName} + AS ( + SELECT note_${tableName}.note_id as id + FROM note_${tableName} + WHERE + ${operator === 'INTERSECT' ? 1 : 0} ${operator} + ${requiredNotesQuery} + )`; + } + + // Get the ROWIDs that satisfy the condition so we can filter the result + const whereCondition = ` + ${relation} ROWID ${(relation === 'AND' && requirement === 'EXCLUSION') ? 'NOT' : ''} + IN ( + SELECT notes_normalized.ROWID + FROM notes_with_${requirement}_${tableName} + JOIN notes_normalized + ON notes_with_${requirement}_${tableName}.id=notes_normalized.id + )`; + + withs.push(withCondition); + params.push(...values); + conditions.push(whereCondition); +}; + + +const resourceFilter = (terms: Term[], conditions: string[], params: string[], relation: Relation, withs: string[]) => { + const tableName = 'resources'; + + const resourceIDs = ` + SELECT resources.id + FROM resources + WHERE resources.mime LIKE ?`; + + const noteIDsWithResource = ` + SELECT note_resources.note_id AS id + FROM note_resources + WHERE note_resources.is_associated=1 + AND note_resources.resource_id IN (${resourceIDs})`; + + const requiredResources = terms.filter(x => x.name === 'resource' && !x.negated); + const excludedResources = terms.filter(x => x.name === 'resource' && x.negated); + + if (requiredResources.length > 0) { + filterByTableName(requiredResources, conditions, params, relation, noteIDsWithResource, Requirement.INCLUSION, withs, tableName); + } + + if (excludedResources.length > 0) { + filterByTableName(excludedResources, conditions, params, relation, noteIDsWithResource, Requirement.EXCLUSION, withs, tableName); + } +}; + +const tagFilter = (terms: Term[], conditions: string[], params: string[], relation: Relation, withs: string[]) => { + const tableName = 'tags'; + + const tagIDs = ` + SELECT tags.id + FROM tags + WHERE tags.title + LIKE ?`; + + const noteIDsWithTag = ` + SELECT note_tags.note_id AS id + FROM note_tags + WHERE note_tags.tag_id IN (${tagIDs})`; + + const requiredTags = terms.filter(x => x.name === 'tag' && !x.negated); + const excludedTags = terms.filter(x => x.name === 'tag' && x.negated); + + if (requiredTags.length > 0) { + filterByTableName(requiredTags, conditions, params, relation, noteIDsWithTag, Requirement.INCLUSION, withs, tableName); + } + + if (excludedTags.length > 0) { + filterByTableName(excludedTags, conditions, params, relation, noteIDsWithTag, Requirement.EXCLUSION, withs, tableName); + } +}; + +const genericFilter = (terms: Term[], conditions: string[], params: string[], relation: Relation, fieldName: string) => { + if (fieldName === 'iscompleted' || fieldName === 'type') { + // Faster query when values can only take two distinct values + biConditionalFilter(terms, conditions, relation, fieldName); + return; + } + + const getCondition = (term: Term) => { + if (fieldName === 'sourceurl') { return `notes_normalized.source_url ${term.negated ? 'NOT' : ''} LIKE ?`; } else { return `notes_normalized.${fieldName === 'date' ? `user_${term.name}_time` : `${term.name}`} ${term.negated ? '<' : '>='} ?`; } + }; + + terms.forEach(term => { + conditions.push(` + ${relation} ROWID IN ( + SELECT ROWID + FROM notes_normalized + WHERE ${getCondition(term)} + )`); + params.push(term.value); + }); +}; + +const biConditionalFilter = (terms: Term[], conditions: string[], relation: Relation, filterName: string) => { + const getCondition = (filterName: string , value: string, relation: Relation) => { + const tableName = (relation === 'AND') ? 'notes_fts' : 'notes_normalized'; + if (filterName === 'type') { + return `${tableName}.is_todo IS ${value === 'todo' ? 1 : 0}`; + } else if (filterName === 'iscompleted') { + return `${tableName}.is_todo IS 1 AND ${tableName}.todo_completed IS ${value === '1' ? 'NOT 0' : '0'}`; + } else { + throw new Error('Invalid filter name.'); + } + }; + + const values = terms.map(x => x.value); + + // AND and OR are handled differently because FTS restricts how OR can be used. + values.forEach(value => { + if (relation === 'AND') { + conditions.push(` + AND ${getCondition(filterName, value, relation)}`); + } + if (relation === 'OR') { + conditions.push(` + OR ROWID IN ( + SELECT ROWID + FROM notes_normalized + WHERE ${getCondition(filterName, value, relation)} + )`); + } + }); +}; + +const typeFilter = (terms: Term[], conditions: string[], params: string[], relation: Relation) => { + const typeTerms = terms.filter(x => x.name === 'type'); + genericFilter(typeTerms, conditions, params, relation, 'type'); +}; + +const completedFilter = (terms: Term[], conditions: string[], params: string[], relation: Relation) => { + const completedTerms = terms.filter(x => x.name === 'iscompleted'); + genericFilter(completedTerms, conditions, params, relation, 'iscompleted'); +}; + + +const locationFilter = (terms: Term[], conditons: string[], params: string[], relation: Relation) => { + const locationTerms = terms.filter(x => x.name === 'latitude' || x.name === 'longitude' || x.name === 'altitude'); + genericFilter(locationTerms, conditons, params, relation, 'location'); +}; + +const dateFilter = (terms: Term[], conditons: string[], params: string[], relation: Relation) => { + const getUnixMs = (date:string): string => { + const yyyymmdd = /^[0-9]{8}$/; + const yyyymm = /^[0-9]{6}$/; + const yyyy = /^[0-9]{4}$/; + const smartValue = /^(day|week|month|year)-([0-9]+)$/i; + + if (yyyymmdd.test(date)) { + return time.formatLocalToMs(date, 'YYYYMMDD').toString(); + } else if (yyyymm.test(date)) { + return time.formatLocalToMs(date, 'YYYYMM').toString(); + } else if (yyyy.test(date)) { + return time.formatLocalToMs(date, 'YYYY').toString(); + } else if (smartValue.test(date)) { + const match = smartValue.exec(date); + const timeUnit = match[1]; // eg. day, week, month, year + const num = Number(match[2]); // eg. 1, 12, 15 + return time.goBackInTime(Date.now(), num, timeUnit); + } else { + throw new Error('Invalid date format!'); + } + }; + + const dateTerms = terms.filter(x => x.name === 'created' || x.name === 'updated'); + const unixDateTerms = dateTerms.map(term => { return { ...term, value: getUnixMs(term.value) }; }); + genericFilter(unixDateTerms, conditons, params, relation, 'date'); +}; + +const sourceUrlFilter = (terms: Term[], conditons: string[], params: string[], relation: Relation) => { + const urlTerms = terms.filter(x => x.name === 'sourceurl'); + genericFilter(urlTerms, conditons, params, relation, 'sourceurl'); +}; + + +const textFilter = (terms: Term[], conditions: string[], params: string[], relation: Relation) => { + const addExcludeTextConditions = (excludedTerms: Term[], conditions:string[], params: string[], relation: Relation) => { + const type = excludedTerms[0].name === 'text' ? '' : `.${excludedTerms[0].name}`; + + if (relation === 'AND') { + conditions.push(` + AND ROWID NOT IN ( + SELECT ROWID + FROM notes_fts + WHERE notes_fts${type} MATCH ? + )`); + params.push(excludedTerms.map(x => x.value).join(' OR ')); + } + + if (relation === 'OR') { + excludedTerms.forEach(term => { + conditions.push(` + OR ROWID IN ( + SELECT * + FROM ( + SELECT ROWID + FROM notes_fts + EXCEPT + SELECT ROWID + FROM notes_fts + WHERE notes_fts${type} MATCH ? + ) + )`); + params.push(term.value); + }); + } + }; + + const allTerms = terms.filter(x => x.name === 'title' || x.name === 'body' || x.name === 'text'); + + const includedTerms = allTerms.filter(x => !x.negated); + if (includedTerms.length > 0) { + conditions.push(`${relation} notes_fts MATCH ?`); + const termsToMatch = includedTerms.map(term => { + if (term.name === 'text') return term.value; + else return `${term.name}:${term.value}`; + }); + const matchQuery = (relation === 'OR') ? termsToMatch.join(' OR ') : termsToMatch.join(' '); + params.push(matchQuery); + } + + const excludedTextTerms = allTerms.filter(x => x.name === 'text' && x.negated); + const excludedTitleTerms = allTerms.filter(x => x.name === 'title' && x.negated); + const excludedBodyTerms = allTerms.filter(x => x.name === 'body' && x.negated); + + if ((excludedTextTerms.length > 0)) { + addExcludeTextConditions(excludedTextTerms, conditions, params, relation); + } + + if (excludedTitleTerms.length > 0) { + addExcludeTextConditions(excludedTitleTerms, conditions, params, relation); + } + + if (excludedBodyTerms.length > 0) { + addExcludeTextConditions(excludedBodyTerms, conditions, params, relation); + } +}; + +const getDefaultRelation = (terms: Term[]): Relation => { + const anyTerm = terms.find(term => term.name === 'any'); + if (anyTerm) { return (anyTerm.value === '1') ? Relation.OR : Relation.AND; } + return Relation.AND; +}; + +const getConnective = (terms: Term[], relation: Relation): string => { + const notebookTerm = terms.find(x => x.name === 'notebook'); + return (!notebookTerm && (relation === 'OR')) ? 'ROWID=-1' : '1'; // ROWID=-1 acts as 0 (something always false) +}; + +export default function queryBuilder(terms: Term[]) { + const queryParts: string[] = []; + const params: string[] = []; + const withs: string[] = []; + + // console.log("testing beep beep boop boop") + // console.log(terms); + + const relation: Relation = getDefaultRelation(terms); + + queryParts.push(` + SELECT + notes_fts.id, + notes_fts.title, + offsets(notes_fts) AS offsets, + notes_fts.user_created_time, + notes_fts.user_updated_time, + notes_fts.is_todo, + notes_fts.todo_completed, + notes_fts.parent_id + FROM notes_fts + WHERE ${getConnective(terms, relation)}`); + + + notebookFilter(terms, queryParts, params, withs); + + tagFilter(terms, queryParts, params, relation, withs); + + resourceFilter(terms, queryParts, params, relation, withs); + + + textFilter(terms, queryParts, params, relation); + + typeFilter(terms, queryParts, params, relation); + + completedFilter(terms, queryParts, params, relation); + + dateFilter(terms, queryParts, params, relation); + + locationFilter(terms, queryParts, params, relation); + + sourceUrlFilter(terms, queryParts, params, relation); + + let query; + if (withs.length > 0) { + query = ['WITH RECURSIVE' , withs.join(',') ,queryParts.join(' ')].join(' '); + } else { + query = queryParts.join(' '); + } + + return { query, params }; +} diff --git a/ReactNativeClient/lib/time-utils.js b/ReactNativeClient/lib/time-utils.js index 0b4232477..e8e81996f 100644 --- a/ReactNativeClient/lib/time-utils.js +++ b/ReactNativeClient/lib/time-utils.js @@ -111,6 +111,17 @@ class Time { sleep(seconds) { return this.msleep(seconds * 1000); } + + + goBackInTime(startDate, n, period) { + // period is a string (eg. "day", "week", "month", "year" ), n is an integer + return moment(startDate).startOf(period).subtract(n, period).format('x'); + } + + goForwardInTime(startDate, n, period) { + return moment(startDate).startOf(period).add(n, period).format('x'); + } + } const time = new Time(); diff --git a/ReactNativeClient/root.js b/ReactNativeClient/root.js index 5d9b1b7b9..478e6a278 100644 --- a/ReactNativeClient/root.js +++ b/ReactNativeClient/root.js @@ -61,7 +61,7 @@ const DropdownAlert = require('react-native-dropdownalert').default; const ShareExtension = require('lib/ShareExtension.js').default; const handleShared = require('lib/shareHandler').default; const ResourceFetcher = require('lib/services/ResourceFetcher'); -const SearchEngine = require('lib/services/SearchEngine'); +const SearchEngine = require('lib/services/searchengine/SearchEngine'); const WelcomeUtils = require('lib/WelcomeUtils'); const { themeStyle } = require('lib/components/global-style.js'); const { uuid } = require('lib/uuid.js'); From 8c7a24282cb1d6f368b580b015909d5f1e4533f5 Mon Sep 17 00:00:00 2001 From: alexchee Date: Fri, 7 Aug 2020 19:30:11 -0400 Subject: [PATCH 2/4] All: Fixes #3591: Fixed sync fetching issue (#3599) --- CliClient/tests/file_api_driver.js | 1 + ReactNativeClient/lib/file-api-driver-amazon-s3.js | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/CliClient/tests/file_api_driver.js b/CliClient/tests/file_api_driver.js index dac48db61..a70b4b49b 100644 --- a/CliClient/tests/file_api_driver.js +++ b/CliClient/tests/file_api_driver.js @@ -50,6 +50,7 @@ const api = null; // const items = response.items; // expect(items.length).toBe(3); // expect(items[0].path).toBe('1'); +// expect(items[0].updated_time).toMatch(/^\d+$/); // make sure it's using epoch timestamp // })); // it('should default to only files on root directory', asyncTest(async () => { diff --git a/ReactNativeClient/lib/file-api-driver-amazon-s3.js b/ReactNativeClient/lib/file-api-driver-amazon-s3.js index 7348fd579..75eaebbb2 100644 --- a/ReactNativeClient/lib/file-api-driver-amazon-s3.js +++ b/ReactNativeClient/lib/file-api-driver-amazon-s3.js @@ -145,10 +145,11 @@ class FileApiDriverAmazonS3 { metadataToStat_(md, path) { const relativePath = basename(path); + const lastModifiedDate = md['LastModified'] ? new Date(md['LastModified']) : new Date(); const output = { path: relativePath, - updated_time: md['LastModified'] ? new Date(md['LastModified']) : new Date(), + updated_time: lastModifiedDate.getTime(), isDeleted: !!md['DeleteMarker'], isDir: false, }; From aa147bbcdc3813e5999c3377fbadff0bfdc80c04 Mon Sep 17 00:00:00 2001 From: R3dError <50834839+R3dError@users.noreply.github.com> Date: Sat, 8 Aug 2020 01:32:49 +0200 Subject: [PATCH 3/4] Desktop: Added attach file to menu bar (#3540) --- ElectronClient/app.js | 1 + 1 file changed, 1 insertion(+) diff --git a/ElectronClient/app.js b/ElectronClient/app.js index 2acdb30e9..bb1556792 100644 --- a/ElectronClient/app.js +++ b/ElectronClient/app.js @@ -756,6 +756,7 @@ class Application extends BaseApplication { cmdService.commandToMenuItem('textCode'), separator(), cmdService.commandToMenuItem('insertDateTime'), + cmdService.commandToMenuItem('attachFile'), separator(), cmdService.commandToMenuItem('focusSearch'), cmdService.commandToMenuItem('showLocalSearch'), From 799a9e810d1b7001ab0020933cf2d25c8ee3467d Mon Sep 17 00:00:00 2001 From: jonath92 <49979415+jonath92@users.noreply.github.com> Date: Sat, 8 Aug 2020 01:35:30 +0200 Subject: [PATCH 4/4] All: Resolves #1266: Add support for OneDrive for Business (#3433) --- ReactNativeClient/lib/SyncTargetOneDrive.js | 20 +++++++++++-- .../lib/file-api-driver-onedrive.js | 20 +++++++------ ReactNativeClient/lib/onedrive-api.js | 29 +++++++++++++++++-- ReactNativeClient/lib/registry.js | 15 ---------- 4 files changed, 56 insertions(+), 28 deletions(-) diff --git a/ReactNativeClient/lib/SyncTargetOneDrive.js b/ReactNativeClient/lib/SyncTargetOneDrive.js index 57a197239..ce6c6a2eb 100644 --- a/ReactNativeClient/lib/SyncTargetOneDrive.js +++ b/ReactNativeClient/lib/SyncTargetOneDrive.js @@ -82,6 +82,15 @@ class SyncTargetOneDrive extends BaseSyncTarget { } async initFileApi() { + let context = Setting.value(`sync.${this.syncTargetId()}.context`); + context = context === '' ? null : JSON.parse(context); + let accountProperties = context ? context.accountProperties : null; + if (!accountProperties) { + accountProperties = await this.api_.execAccountPropertiesRequest(); + context ? context.accountProperties = accountProperties : context = { accountProperties: accountProperties }; + Setting.setValue(`sync.${this.syncTargetId()}.context`, JSON.stringify(context)); + } + this.api_.setAccountProperties(accountProperties); const appDir = await this.api().appDirectory(); const fileApi = new FileApi(appDir, new FileApiDriverOneDrive(this.api())); fileApi.setSyncTargetId(this.syncTargetId()); @@ -90,8 +99,15 @@ class SyncTargetOneDrive extends BaseSyncTarget { } async initSynchronizer() { - if (!(await this.isAuthenticated())) throw new Error('User is not authentified'); - return new Synchronizer(this.db(), await this.fileApi(), Setting.value('appType')); + try { + if (!(await this.isAuthenticated())) throw new Error('User is not authentified'); + return new Synchronizer(this.db(), await this.fileApi(), Setting.value('appType')); + } catch (error) { + BaseSyncTarget.dispatch({ type: 'SYNC_REPORT_UPDATE', report: { errors: [error] } }); + throw error; + } + + } } diff --git a/ReactNativeClient/lib/file-api-driver-onedrive.js b/ReactNativeClient/lib/file-api-driver-onedrive.js index bf5ecf94c..a7d939f1a 100644 --- a/ReactNativeClient/lib/file-api-driver-onedrive.js +++ b/ReactNativeClient/lib/file-api-driver-onedrive.js @@ -217,7 +217,9 @@ class FileApiDriverOneDrive { }; const freshStartDelta = () => { - const url = `${this.makePath_(path)}:/delta`; + // Business Accounts are only allowed to make delta requests to the root. For some reason /delta gives an error for personal accounts and :/delta an error for business accounts + const accountProperties = this.api_.accountProperties_; + const url = (accountProperties.accountType === 'business') ? `/drives/${accountProperties.driveId}/root/delta` : `${this.makePath_(path)}:/delta`; const query = this.itemFilter_(); query.select += ',deleted'; return { url: url, query: query }; @@ -265,14 +267,14 @@ class FileApiDriverOneDrive { const items = []; - // The delta API might return things that happen in subdirectories of the root and we don't want to - // deal with these since all the files we're interested in are at the root (The .resource dir - // is special since it's managed directly by the clients and resources never change - only the - // associated .md file at the root is synced). So in the loop below we check that the parent is - // indeed the root, otherwise the item is skipped. - // (Not sure but it's possible the delta API also returns events for files that are copied outside - // of the app directory and later deleted or modified. We also don't want to deal with - // these files during sync). + // The delta API might return things that happens in subdirectories and outside of the joplin directory. + // We don't want to deal with these since all the files we're interested in are at the root of the joplin directory + // (The .resource dir is special since it's managed directly by the clients and resources never change - only the + // associated .md file at the root is synced). So in the loop below we check that the parent is indeed the joplin + // directory, otherwise the item is skipped. + // At OneDrive for Business delta requests can only make at the root of OneDrive. Not sure but it's possible that + // the delta API also returns events for files that are copied outside of the app directory and later deleted or + // modified when using OneDrive Personal). for (let i = 0; i < response.value.length; i++) { const v = response.value[i]; diff --git a/ReactNativeClient/lib/onedrive-api.js b/ReactNativeClient/lib/onedrive-api.js index e67adcf46..33b41863c 100644 --- a/ReactNativeClient/lib/onedrive-api.js +++ b/ReactNativeClient/lib/onedrive-api.js @@ -13,6 +13,7 @@ class OneDriveApi { this.clientId_ = clientId; this.clientSecret_ = clientSecret; this.auth_ = null; + this.accountProperties_ = null; this.isPublic_ = isPublic; this.listeners_ = { authRefreshed: [], @@ -73,14 +74,15 @@ class OneDriveApi { } async appDirectory() { - const r = await this.execJson('GET', '/drive/special/approot'); + const driveId = this.accountProperties_.driveId; + const r = await this.execJson('GET', `/me/drives/${driveId}/special/approot`); return `${r.parentReference.path}/${r.name}`; } authCodeUrl(redirectUri) { const query = { client_id: this.clientId_, - scope: 'files.readwrite offline_access', + scope: 'files.readwrite offline_access sites.readwrite.all', response_type: 'code', redirect_uri: redirectUri, }; @@ -320,6 +322,29 @@ class OneDriveApi { throw new Error(`Could not execute request after multiple attempts: ${method} ${url}`); } + setAccountProperties(accountProperties) { + this.accountProperties_ = accountProperties; + } + + async execAccountPropertiesRequest() { + const response = await shim.fetch('https://graph.microsoft.com/v1.0/me/drive', { + method: 'GET', + headers: { + 'Authorization': this.token(), + }, + }); + + if (!response.ok) { + const text = await response.text(); + throw new Error(`Could not retrieve account details (drive ID, Account type): ${response.status}: ${response.statusText}: ${text}`); + } else { + const data = await response.json(); + const accountProperties = { accountType: data.driveType, driveId: data.id }; + return accountProperties; + } + + } + async execJson(method, path, query, data) { const response = await this.exec(method, path, query, data); const errorResponseText = await response.text(); diff --git a/ReactNativeClient/lib/registry.js b/ReactNativeClient/lib/registry.js index 28d5bded5..eb0d93c2b 100644 --- a/ReactNativeClient/lib/registry.js +++ b/ReactNativeClient/lib/registry.js @@ -2,7 +2,6 @@ const { Logger } = require('lib/logger.js'); const Setting = require('lib/models/Setting.js'); const { shim } = require('lib/shim.js'); const SyncTargetRegistry = require('lib/SyncTargetRegistry.js'); -const { _ } = require('lib/locale.js'); const reg = {}; @@ -141,20 +140,6 @@ reg.scheduleSync = async (delay = null, syncOptions = null) => { } catch (error) { reg.logger().info('Could not run background sync:'); reg.logger().info(error); - - // Special case to display OneDrive Business error. This is the full error that's received when trying to use a OneDrive Business account: - // - // {"error":"invalid_client","error_description":"AADSTS50011: The reply address 'http://localhost:1917' does not match the reply addresses configured for - // the application: 'cbabb902-d276-4ea4-aa88-062a5889d6dc'. More details: not specified\r\nTrace ID: 6e63dac6-8b37-47e2-bd1b-4768f8713400\r\nCorrelation - // ID: acfd6503-8d97-4349-ae2e-e7a19dd7b6bc\r\nTimestamp: 2017-12-01 13:35:55Z","error_codes":[50011],"timestamp":"2017-12-01 13:35:55Z","trace_id": - // "6e63dac6-8b37-47e2-bd1b-4768f8713400","correlation_id":"acfd6503-8d97-4349-ae2e-e7a19dd7b6bc"}: TOKEN: null Error: {"error":"invalid_client", - // "error_description":"AADSTS50011: The reply address 'http://localhost:1917' does not match the reply addresses configured for the application: - // 'cbabb902-d276-4ea4-aa88-062a5889d6dc'. More details: not specified\r\nTrace ID: 6e63dac6-8b37-47e2-bd1b-4768f8713400\r\nCorrelation ID - // acfd6503-8d97-4349-ae2e-e7a19dd7b6bc\r\nTimestamp: 2017-12-01 13:35:55Z","error_codes":[50011],"timestamp":"2017-12-01 13:35:55Z","trace_id": - // "6e63dac6-8b37-47e2-bd1b-4768f8713400","correlation_id":"acfd6503-8d97-4349-ae2e-e7a19dd7b6bc"} - if (error && error.message && error.message.indexOf('"invalid_client"') >= 0) { - reg.showErrorMessageBox(_('Could not synchronise with OneDrive.\n\nThis error often happens when using OneDrive for Business, which unfortunately cannot be supported.\n\nPlease consider using a regular OneDrive account.')); - } } reg.setupRecurrentSync(); promiseResolve();