1
0
mirror of https://github.com/bpatrik/pigallery2.git synced 2025-01-28 05:36:45 +02:00
pigallery2/backend/model/sql/SearchManager.ts

195 lines
6.7 KiB
TypeScript
Raw Normal View History

2016-12-28 11:50:05 +01:00
import {AutoCompleteItem, SearchTypes} from "../../../common/entities/AutoCompleteItem";
import {ISearchManager} from "../interfaces/ISearchManager";
import {SearchResultDTO} from "../../../common/entities/SearchResultDTO";
import {SQLConnection} from "./SQLConnection";
2016-12-28 11:50:05 +01:00
import {PhotoEntity} from "./enitites/PhotoEntity";
import {DirectoryEntity} from "./enitites/DirectoryEntity";
2016-12-28 11:50:05 +01:00
import {PositionMetaData} from "../../../common/entities/PhotoDTO";
export class SearchManager implements ISearchManager {
2017-07-08 00:18:24 +02:00
async autocomplete(text: string) {
const connection = await SQLConnection.getConnection();
2017-07-08 00:18:24 +02:00
let result: Array<AutoCompleteItem> = [];
let photoRepository = connection.getRepository(PhotoEntity);
let directoryRepository = connection.getRepository(DirectoryEntity);
(await photoRepository
.createQueryBuilder('photo')
.select('DISTINCT(photo.metadataKeywords)')
.where('photo.metadata.keywords LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
2017-10-19 12:08:07 -04:00
.limit(5)
2017-07-08 00:18:24 +02:00
.getRawMany())
.map(r => <Array<string>>JSON.parse(r.metadataKeywords))
.forEach(keywords => {
result = result.concat(this.encapsulateAutoComplete(keywords.filter(k => k.toLowerCase().indexOf(text.toLowerCase()) != -1), SearchTypes.keyword));
});
(await photoRepository
.createQueryBuilder('photo')
.select('DISTINCT(photo.metadataPositionData)')
.where('photo.metadata.positionData LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
2017-10-19 12:08:07 -04:00
.limit(5)
2017-07-08 00:18:24 +02:00
.getRawMany())
.map(r => <PositionMetaData>JSON.parse(r.metadataPositionData))
.filter(pm => !!pm)
.map(pm => <Array<string>>[pm.city || "", pm.country || "", pm.state || ""])
.forEach(positions => {
result = result.concat(this.encapsulateAutoComplete(positions.filter(p => p.toLowerCase().indexOf(text.toLowerCase()) != -1), SearchTypes.position));
});
result = result.concat(this.encapsulateAutoComplete((await photoRepository
.createQueryBuilder('photo')
.select('DISTINCT(photo.name)')
.where('photo.name LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
2017-10-19 12:08:07 -04:00
.limit(5)
2017-07-08 00:18:24 +02:00
.getRawMany())
.map(r => r.name), SearchTypes.image));
result = result.concat(this.encapsulateAutoComplete((await directoryRepository
.createQueryBuilder('dir')
.select('DISTINCT(dir.name)')
.where('dir.name LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
2017-10-19 12:08:07 -04:00
.limit(5)
2017-07-08 00:18:24 +02:00
.getRawMany())
.map(r => r.name), SearchTypes.directory));
return this.autoCompleteItemsUnique(result);
2017-07-07 22:54:18 +02:00
}
2017-07-25 21:36:28 +02:00
async search(text: string, searchType: SearchTypes) {
const connection = await SQLConnection.getConnection();
2017-07-07 22:54:18 +02:00
2017-07-08 00:18:24 +02:00
let result: SearchResultDTO = <SearchResultDTO>{
searchText: text,
searchType: searchType,
directories: [],
2017-07-25 21:36:28 +02:00
photos: [],
resultOverflow: false
2017-07-08 00:18:24 +02:00
};
2017-07-07 22:54:18 +02:00
2017-07-08 00:18:24 +02:00
let query = connection
.getRepository(PhotoEntity)
.createQueryBuilder("photo")
.innerJoinAndSelect("photo.directory", "directory")
.orderBy("photo.metadata.creationDate", "ASC");
2017-07-07 22:54:18 +02:00
2017-07-08 00:18:24 +02:00
if (!searchType || searchType === SearchTypes.directory) {
query.orWhere('directory.name LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"});
}
2017-07-07 22:54:18 +02:00
2017-07-08 00:18:24 +02:00
if (!searchType || searchType === SearchTypes.image) {
query.orWhere('photo.name LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"});
}
if (!searchType || searchType === SearchTypes.position) {
query.orWhere('photo.metadata.positionData LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"});
}
if (!searchType || searchType === SearchTypes.keyword) {
query.orWhere('photo.metadata.keywords LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"});
}
let photos = await query
2017-10-19 12:08:07 -04:00
.limit(2001)
2017-07-08 00:18:24 +02:00
.getMany();
if (photos) {
for (let i = 0; i < photos.length; i++) {
photos[i].metadata.keywords = <any>JSON.parse(<any>photos[i].metadata.keywords);
photos[i].metadata.cameraData = <any>JSON.parse(<any>photos[i].metadata.cameraData);
photos[i].metadata.positionData = <any>JSON.parse(<any>photos[i].metadata.positionData);
photos[i].metadata.size = <any>JSON.parse(<any>photos[i].metadata.size);
2017-07-07 22:54:18 +02:00
}
2017-07-08 00:18:24 +02:00
result.photos = photos;
2017-07-25 21:36:28 +02:00
if (result.photos.length > 2000) {
result.resultOverflow = true;
}
2017-07-08 00:18:24 +02:00
}
2017-07-07 22:54:18 +02:00
2017-07-08 00:18:24 +02:00
result.directories = await connection
.getRepository(DirectoryEntity)
.createQueryBuilder("dir")
.where('dir.name LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
2017-10-19 12:08:07 -04:00
.limit(201)
2017-07-08 00:18:24 +02:00
.getMany();
2017-07-07 22:54:18 +02:00
2017-07-25 21:36:28 +02:00
if (result.directories.length > 200) {
result.resultOverflow = true;
}
2017-07-07 22:54:18 +02:00
2017-07-08 00:18:24 +02:00
return result;
2017-07-07 22:54:18 +02:00
}
2017-07-25 21:36:28 +02:00
async instantSearch(text: string) {
const connection = await SQLConnection.getConnection();
2017-07-08 00:18:24 +02:00
let result: SearchResultDTO = <SearchResultDTO>{
searchText: text,
directories: [],
2017-07-25 21:36:28 +02:00
photos: [],
resultOverflow: false
2017-07-08 00:18:24 +02:00
};
let photos = await connection
.getRepository(PhotoEntity)
.createQueryBuilder("photo")
.orderBy("photo.metadata.creationDate", "ASC")
.where('photo.metadata.keywords LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
.orWhere('photo.metadata.positionData LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
.orWhere('photo.name LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
.innerJoinAndSelect("photo.directory", "directory")
2017-10-19 12:08:07 -04:00
.limit(10)
2017-07-08 00:18:24 +02:00
.getMany();
if (photos) {
for (let i = 0; i < photos.length; i++) {
photos[i].metadata.keywords = <any>JSON.parse(<any>photos[i].metadata.keywords);
photos[i].metadata.cameraData = <any>JSON.parse(<any>photos[i].metadata.cameraData);
photos[i].metadata.positionData = <any>JSON.parse(<any>photos[i].metadata.positionData);
photos[i].metadata.size = <any>JSON.parse(<any>photos[i].metadata.size);
2017-07-07 22:54:18 +02:00
}
2017-07-08 00:18:24 +02:00
result.photos = photos;
}
2017-07-07 22:54:18 +02:00
2017-07-30 09:06:12 +02:00
const directories = await connection
2017-07-08 00:18:24 +02:00
.getRepository(DirectoryEntity)
.createQueryBuilder("dir")
.where('dir.name LIKE :text COLLATE utf8_general_ci', {text: "%" + text + "%"})
2017-10-19 12:08:07 -04:00
.limit(10)
2017-07-08 00:18:24 +02:00
.getMany();
2017-07-07 22:54:18 +02:00
2017-07-08 00:18:24 +02:00
result.directories = directories;
2017-07-07 22:54:18 +02:00
2017-07-08 00:18:24 +02:00
return result;
2017-07-07 22:54:18 +02:00
}
private encapsulateAutoComplete(values: Array<string>, type: SearchTypes) {
let res = [];
values.forEach((value) => {
res.push(new AutoCompleteItem(value, type));
});
return res;
}
private autoCompleteItemsUnique(array: Array<AutoCompleteItem>) {
let a = array.concat();
for (let i = 0; i < a.length; ++i) {
for (let j = i + 1; j < a.length; ++j) {
if (a[i].equals(a[j]))
a.splice(j--, 1);
}
2016-12-28 11:50:05 +01:00
}
2017-07-07 22:54:18 +02:00
return a;
}
}