2018-03-30 21:30:30 +02:00
|
|
|
import {expect} from 'chai';
|
|
|
|
import {SQLConnection} from '../../../../../backend/model/sql/SQLConnection';
|
2019-01-27 01:03:40 +02:00
|
|
|
import {PhotoEntity} from '../../../../../backend/model/sql/enitites/PhotoEntity';
|
2018-03-30 21:30:30 +02:00
|
|
|
import {SearchManager} from '../../../../../backend/model/sql/SearchManager';
|
|
|
|
import {AutoCompleteItem, SearchTypes} from '../../../../../common/entities/AutoCompleteItem';
|
|
|
|
import {SearchResultDTO} from '../../../../../common/entities/SearchResultDTO';
|
|
|
|
import {DirectoryEntity} from '../../../../../backend/model/sql/enitites/DirectoryEntity';
|
2018-11-17 23:46:34 +02:00
|
|
|
import {Utils} from '../../../../../common/Utils';
|
2018-11-24 12:50:11 +02:00
|
|
|
import {TestHelper} from './TestHelper';
|
2018-11-24 15:51:47 +02:00
|
|
|
import {VideoEntity} from '../../../../../backend/model/sql/enitites/VideoEntity';
|
2019-01-27 01:03:40 +02:00
|
|
|
import {PersonEntry} from '../../../../../backend/model/sql/enitites/PersonEntry';
|
|
|
|
import {FaceRegionEntry} from '../../../../../backend/model/sql/enitites/FaceRegionEntry';
|
|
|
|
import {PhotoDTO} from '../../../../../common/entities/PhotoDTO';
|
2019-01-27 21:36:42 +02:00
|
|
|
import {SQLTestHelper} from '../../../SQLTestHelper';
|
|
|
|
import {Config} from '../../../../../common/config/private/Config';
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2019-01-27 21:36:42 +02:00
|
|
|
// to help WebStorm to handle the test cases
|
|
|
|
declare let describe: any;
|
|
|
|
declare const after: any;
|
|
|
|
describe = SQLTestHelper.describe;
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2019-01-27 21:36:42 +02:00
|
|
|
describe('SearchManager', (sqlHelper: SQLTestHelper) => {
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2018-11-24 12:50:11 +02:00
|
|
|
const dir = TestHelper.getDirectoryEntry();
|
|
|
|
const p = TestHelper.getPhotoEntry1(dir);
|
|
|
|
const p2 = TestHelper.getPhotoEntry2(dir);
|
2019-01-27 01:03:40 +02:00
|
|
|
const p_faceLess = TestHelper.getPhotoEntry2(dir);
|
|
|
|
delete p_faceLess.metadata.faces;
|
|
|
|
p_faceLess.name = 'fl';
|
2018-11-24 15:51:47 +02:00
|
|
|
const v = TestHelper.getVideoEntry1(dir);
|
2018-01-31 03:01:16 +02:00
|
|
|
|
|
|
|
const setUpSqlDB = async () => {
|
2019-01-27 21:36:42 +02:00
|
|
|
await sqlHelper.initDB();
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2019-01-27 01:03:40 +02:00
|
|
|
const savePhoto = async (photo: PhotoDTO) => {
|
|
|
|
const savedPhoto = await pr.save(photo);
|
|
|
|
if (!photo.metadata.faces) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (let i = 0; i < photo.metadata.faces.length; i++) {
|
|
|
|
const face = photo.metadata.faces[i];
|
|
|
|
const person = await conn.getRepository(PersonEntry).save({name: face.name});
|
|
|
|
await conn.getRepository(FaceRegionEntry).save({box: face.box, person: person, media: savedPhoto});
|
|
|
|
}
|
|
|
|
};
|
2018-01-31 03:01:16 +02:00
|
|
|
const conn = await SQLConnection.getConnection();
|
|
|
|
|
|
|
|
const pr = conn.getRepository(PhotoEntity);
|
|
|
|
|
|
|
|
await conn.getRepository(DirectoryEntity).save(p.directory);
|
2019-01-27 01:03:40 +02:00
|
|
|
await savePhoto(p);
|
|
|
|
await savePhoto(p2);
|
|
|
|
await savePhoto(p_faceLess);
|
|
|
|
|
2018-11-24 15:51:47 +02:00
|
|
|
await conn.getRepository(VideoEntity).save(v);
|
2018-01-31 03:01:16 +02:00
|
|
|
|
|
|
|
await SQLConnection.close();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
await setUpSqlDB();
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2019-01-27 21:36:42 +02:00
|
|
|
after(async () => {
|
|
|
|
await sqlHelper.clearDB();
|
|
|
|
});
|
2018-01-31 03:01:16 +02:00
|
|
|
|
|
|
|
it('should get autocomplete', async () => {
|
2018-11-02 12:22:18 +02:00
|
|
|
const sm = new SearchManager();
|
2018-01-31 03:01:16 +02:00
|
|
|
|
|
|
|
const cmp = (a: AutoCompleteItem, b: AutoCompleteItem) => {
|
2019-01-27 01:03:40 +02:00
|
|
|
if (a.text === b.text) {
|
|
|
|
return a.type - b.type;
|
|
|
|
}
|
2018-01-31 03:01:16 +02:00
|
|
|
return a.text.localeCompare(b.text);
|
|
|
|
};
|
|
|
|
|
2018-03-30 21:30:30 +02:00
|
|
|
expect((await sm.autocomplete('tat'))).to.deep.equal([new AutoCompleteItem('Tatooine', SearchTypes.position)]);
|
|
|
|
expect((await sm.autocomplete('star'))).to.deep.equal([new AutoCompleteItem('star wars', SearchTypes.keyword),
|
|
|
|
new AutoCompleteItem('death star', SearchTypes.keyword)]);
|
|
|
|
|
|
|
|
expect((await sm.autocomplete('wars'))).to.deep.equal([new AutoCompleteItem('star wars', SearchTypes.keyword),
|
|
|
|
new AutoCompleteItem('wars dir', SearchTypes.directory)]);
|
|
|
|
|
|
|
|
expect((await sm.autocomplete('arch'))).eql([new AutoCompleteItem('Research City', SearchTypes.position)]);
|
2019-01-27 21:36:42 +02:00
|
|
|
|
|
|
|
Config.Client.Search.AutoComplete.maxItemsPerCategory = 99999;
|
2018-03-30 21:30:30 +02:00
|
|
|
expect((await sm.autocomplete('a')).sort(cmp)).eql([
|
|
|
|
new AutoCompleteItem('Boba Fett', SearchTypes.keyword),
|
2019-01-27 01:03:40 +02:00
|
|
|
new AutoCompleteItem('Boba Fett', SearchTypes.person),
|
2018-03-30 21:30:30 +02:00
|
|
|
new AutoCompleteItem('star wars', SearchTypes.keyword),
|
|
|
|
new AutoCompleteItem('Anakin', SearchTypes.keyword),
|
2019-01-27 01:03:40 +02:00
|
|
|
new AutoCompleteItem('Anakin Skywalker', SearchTypes.person),
|
|
|
|
new AutoCompleteItem('Luke Skywalker', SearchTypes.person),
|
|
|
|
new AutoCompleteItem('Han Solo', SearchTypes.person),
|
2018-03-30 21:30:30 +02:00
|
|
|
new AutoCompleteItem('death star', SearchTypes.keyword),
|
2019-01-27 01:03:40 +02:00
|
|
|
new AutoCompleteItem('Padmé Amidala', SearchTypes.person),
|
2019-01-27 21:36:42 +02:00
|
|
|
new AutoCompleteItem('Obivan Kenobi', SearchTypes.person),
|
2018-03-30 21:30:30 +02:00
|
|
|
new AutoCompleteItem('Padmé Amidala', SearchTypes.keyword),
|
|
|
|
new AutoCompleteItem('Natalie Portman', SearchTypes.keyword),
|
2018-12-05 20:15:24 +02:00
|
|
|
new AutoCompleteItem('Han Solo\'s dice', SearchTypes.photo),
|
2018-03-30 21:30:30 +02:00
|
|
|
new AutoCompleteItem('Kamino', SearchTypes.position),
|
|
|
|
new AutoCompleteItem('Tatooine', SearchTypes.position),
|
|
|
|
new AutoCompleteItem('wars dir', SearchTypes.directory),
|
|
|
|
new AutoCompleteItem('Research City', SearchTypes.position)].sort(cmp));
|
|
|
|
|
2019-01-27 21:36:42 +02:00
|
|
|
Config.Client.Search.AutoComplete.maxItemsPerCategory = 1;
|
|
|
|
expect((await sm.autocomplete('a')).sort(cmp)).eql([
|
|
|
|
new AutoCompleteItem('Anakin', SearchTypes.keyword),
|
|
|
|
new AutoCompleteItem('star wars', SearchTypes.keyword),
|
|
|
|
new AutoCompleteItem('death star', SearchTypes.keyword),
|
|
|
|
new AutoCompleteItem('Anakin Skywalker', SearchTypes.person),
|
|
|
|
new AutoCompleteItem('Han Solo\'s dice', SearchTypes.photo),
|
|
|
|
new AutoCompleteItem('Kamino', SearchTypes.position),
|
|
|
|
new AutoCompleteItem('Research City', SearchTypes.position),
|
|
|
|
new AutoCompleteItem('wars dir', SearchTypes.directory),
|
|
|
|
new AutoCompleteItem('Boba Fett', SearchTypes.keyword)].sort(cmp));
|
|
|
|
Config.Client.Search.AutoComplete.maxItemsPerCategory = 5;
|
|
|
|
|
2018-11-24 15:51:47 +02:00
|
|
|
expect((await sm.autocomplete('sw')).sort(cmp)).to.deep.equal([new AutoCompleteItem('sw1', SearchTypes.photo),
|
|
|
|
new AutoCompleteItem('sw2', SearchTypes.photo), new AutoCompleteItem(v.name, SearchTypes.video)].sort(cmp));
|
|
|
|
|
|
|
|
expect((await sm.autocomplete(v.name)).sort(cmp)).to.deep.equal([new AutoCompleteItem(v.name, SearchTypes.video)]);
|
2019-01-27 21:36:42 +02:00
|
|
|
|
2018-01-31 03:01:16 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it('should search', async () => {
|
2018-11-02 12:22:18 +02:00
|
|
|
const sm = new SearchManager();
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
|
|
|
|
expect(Utils.clone(await sm.search('sw', null))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'sw',
|
2018-01-31 03:01:16 +02:00
|
|
|
searchType: null,
|
|
|
|
directories: [],
|
2018-11-26 01:26:29 +02:00
|
|
|
media: [p, p2, v],
|
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2019-01-27 01:03:40 +02:00
|
|
|
expect(Utils.clone(await sm.search('Boba', null))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
|
|
|
searchText: 'Boba',
|
|
|
|
searchType: null,
|
|
|
|
directories: [],
|
|
|
|
media: [p],
|
|
|
|
metaFile: [],
|
|
|
|
resultOverflow: false
|
|
|
|
}));
|
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
expect(Utils.clone(await sm.search('Tatooine', SearchTypes.position))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'Tatooine',
|
2018-01-31 03:01:16 +02:00
|
|
|
searchType: SearchTypes.position,
|
|
|
|
directories: [],
|
2018-11-04 20:28:32 +02:00
|
|
|
media: [p],
|
2018-11-26 01:26:29 +02:00
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
expect(Utils.clone(await sm.search('ortm', SearchTypes.keyword))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'ortm',
|
2018-01-31 03:01:16 +02:00
|
|
|
searchType: SearchTypes.keyword,
|
|
|
|
directories: [],
|
2019-01-27 01:03:40 +02:00
|
|
|
media: [p2, p_faceLess],
|
2018-11-26 01:26:29 +02:00
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
expect(Utils.clone(await sm.search('ortm', SearchTypes.keyword))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'ortm',
|
2018-01-31 03:01:16 +02:00
|
|
|
searchType: SearchTypes.keyword,
|
|
|
|
directories: [],
|
2019-01-27 01:03:40 +02:00
|
|
|
media: [p2, p_faceLess],
|
2018-11-26 01:26:29 +02:00
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
expect(Utils.clone(await sm.search('wa', SearchTypes.keyword))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'wa',
|
2018-01-31 03:01:16 +02:00
|
|
|
searchType: SearchTypes.keyword,
|
|
|
|
directories: [dir],
|
2019-01-27 01:03:40 +02:00
|
|
|
media: [p, p2, p_faceLess],
|
2018-11-26 01:26:29 +02:00
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-12-05 20:15:24 +02:00
|
|
|
|
|
|
|
expect(Utils.clone(await sm.search('han', SearchTypes.photo))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
|
|
|
searchText: 'han',
|
|
|
|
searchType: SearchTypes.photo,
|
|
|
|
directories: [],
|
|
|
|
media: [p],
|
|
|
|
metaFile: [],
|
|
|
|
resultOverflow: false
|
|
|
|
}));
|
|
|
|
|
2019-01-27 01:03:40 +02:00
|
|
|
expect(Utils.clone(await sm.search('sw', SearchTypes.video))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
|
|
|
searchText: 'sw',
|
|
|
|
searchType: SearchTypes.video,
|
|
|
|
directories: [],
|
|
|
|
media: [v],
|
|
|
|
metaFile: [],
|
|
|
|
resultOverflow: false
|
|
|
|
}));
|
|
|
|
|
2018-12-05 20:15:24 +02:00
|
|
|
expect(Utils.clone(await sm.search('han', SearchTypes.keyword))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
|
|
|
searchText: 'han',
|
|
|
|
searchType: SearchTypes.keyword,
|
|
|
|
directories: [],
|
|
|
|
media: [],
|
|
|
|
metaFile: [],
|
|
|
|
resultOverflow: false
|
|
|
|
}));
|
2019-01-27 01:03:40 +02:00
|
|
|
|
|
|
|
expect(Utils.clone(await sm.search('Boba', SearchTypes.person))).to.deep.equal(Utils.clone(<SearchResultDTO>{
|
|
|
|
searchText: 'Boba',
|
|
|
|
searchType: SearchTypes.person,
|
|
|
|
directories: [],
|
|
|
|
media: [p],
|
|
|
|
metaFile: [],
|
|
|
|
resultOverflow: false
|
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it('should instant search', async () => {
|
2018-11-02 12:22:18 +02:00
|
|
|
const sm = new SearchManager();
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
expect(Utils.clone(await sm.instantSearch('sw'))).to.deep.equal(Utils.clone({
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'sw',
|
2018-01-31 03:01:16 +02:00
|
|
|
directories: [],
|
2018-11-26 01:26:29 +02:00
|
|
|
media: [p, p2, v],
|
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
expect(Utils.clone(await sm.instantSearch('Tatooine'))).to.deep.equal(Utils.clone({
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'Tatooine',
|
2018-01-31 03:01:16 +02:00
|
|
|
directories: [],
|
2018-11-17 23:46:34 +02:00
|
|
|
media: [p],
|
2018-11-26 01:26:29 +02:00
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
expect(Utils.clone(await sm.instantSearch('ortm'))).to.deep.equal(Utils.clone({
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'ortm',
|
2018-01-31 03:01:16 +02:00
|
|
|
directories: [],
|
2019-01-27 01:03:40 +02:00
|
|
|
media: [p2, p_faceLess],
|
2018-11-26 01:26:29 +02:00
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
|
|
|
|
|
2018-11-17 23:46:34 +02:00
|
|
|
expect(Utils.clone(await sm.instantSearch('wa'))).to.deep.equal(Utils.clone({
|
2018-03-30 21:30:30 +02:00
|
|
|
searchText: 'wa',
|
2018-01-31 03:01:16 +02:00
|
|
|
directories: [dir],
|
2019-01-27 01:03:40 +02:00
|
|
|
media: [p, p2, p_faceLess],
|
2018-11-26 01:26:29 +02:00
|
|
|
metaFile: [],
|
2018-01-31 03:01:16 +02:00
|
|
|
resultOverflow: false
|
2018-11-17 23:46:34 +02:00
|
|
|
}));
|
2018-12-05 20:15:24 +02:00
|
|
|
|
|
|
|
expect(Utils.clone(await sm.instantSearch('han'))).to.deep.equal(Utils.clone({
|
|
|
|
searchText: 'han',
|
|
|
|
directories: [],
|
|
|
|
media: [p],
|
|
|
|
metaFile: [],
|
|
|
|
resultOverflow: false
|
|
|
|
}));
|
2019-01-27 01:03:40 +02:00
|
|
|
expect(Utils.clone(await sm.instantSearch('Boba'))).to.deep.equal(Utils.clone({
|
|
|
|
searchText: 'Boba',
|
|
|
|
directories: [],
|
|
|
|
media: [p],
|
|
|
|
metaFile: [],
|
|
|
|
resultOverflow: false
|
|
|
|
}));
|
2018-01-31 03:01:16 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
});
|