2022-08-06 06:21:11 +02:00
|
|
|
import 'package:drift/drift.dart' as sql;
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/account.dart';
|
2022-09-07 19:01:50 +02:00
|
|
|
import 'package:nc_photos/ci_string.dart';
|
|
|
|
import 'package:nc_photos/di_container.dart';
|
|
|
|
import 'package:nc_photos/entity/file.dart';
|
2022-10-15 16:29:18 +02:00
|
|
|
import 'package:nc_photos/entity/file_descriptor.dart';
|
2022-08-06 06:21:11 +02:00
|
|
|
import 'package:nc_photos/entity/search.dart';
|
2022-09-07 19:01:50 +02:00
|
|
|
import 'package:nc_photos/entity/search_util.dart' as search_util;
|
2023-02-20 15:21:35 +01:00
|
|
|
import 'package:nc_photos/entity/sqlite/database.dart' as sql;
|
|
|
|
import 'package:nc_photos/entity/sqlite/files_query_builder.dart' as sql;
|
|
|
|
import 'package:nc_photos/entity/sqlite/type_converter.dart';
|
2022-09-07 19:01:50 +02:00
|
|
|
import 'package:nc_photos/iterable_extension.dart';
|
2022-08-06 06:21:11 +02:00
|
|
|
import 'package:nc_photos/object_extension.dart';
|
2022-09-07 19:01:50 +02:00
|
|
|
import 'package:nc_photos/use_case/list_tagged_file.dart';
|
|
|
|
import 'package:nc_photos/use_case/populate_person.dart';
|
2022-12-16 16:01:04 +01:00
|
|
|
import 'package:np_codegen/np_codegen.dart';
|
2022-08-06 06:21:11 +02:00
|
|
|
|
2022-12-16 16:01:04 +01:00
|
|
|
part 'data_source.g.dart';
|
|
|
|
|
|
|
|
@npLog
|
2022-08-06 06:21:11 +02:00
|
|
|
class SearchSqliteDbDataSource implements SearchDataSource {
|
2022-09-07 19:01:50 +02:00
|
|
|
SearchSqliteDbDataSource(this._c);
|
2022-08-06 06:21:11 +02:00
|
|
|
|
|
|
|
@override
|
|
|
|
list(Account account, SearchCriteria criteria) async {
|
|
|
|
_log.info("[list] $criteria");
|
2022-09-07 19:01:50 +02:00
|
|
|
final stopwatch = Stopwatch()..start();
|
|
|
|
try {
|
|
|
|
final keywords = search_util
|
|
|
|
.cleanUpSymbols(criteria.input)
|
|
|
|
.split(" ")
|
|
|
|
.where((s) => s.isNotEmpty)
|
|
|
|
.map((s) => s.toCi().toCaseInsensitiveString())
|
|
|
|
.toSet();
|
|
|
|
final futures = await Future.wait([
|
|
|
|
_listByPath(account, criteria, keywords),
|
|
|
|
_listByLocation(account, criteria),
|
|
|
|
_listByTag(account, criteria),
|
|
|
|
_listByPerson(account, criteria),
|
|
|
|
]);
|
|
|
|
return futures.flatten().distinctIf(
|
|
|
|
(a, b) => a.compareServerIdentity(b),
|
|
|
|
(a) => a.identityHashCode,
|
|
|
|
);
|
|
|
|
} finally {
|
|
|
|
_log.info("[list] Elapsed time: ${stopwatch.elapsedMilliseconds}ms");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<List<File>> _listByPath(
|
|
|
|
Account account, SearchCriteria criteria, Set<String> keywords) async {
|
|
|
|
try {
|
|
|
|
final dbFiles = await _c.sqliteDb.use((db) async {
|
|
|
|
final query = db.queryFiles().run((q) {
|
|
|
|
q.setQueryMode(sql.FilesQueryMode.completeFile);
|
|
|
|
q.setAppAccount(account);
|
|
|
|
for (final r in account.roots) {
|
|
|
|
if (r.isNotEmpty) {
|
|
|
|
q.byOrRelativePathPattern("$r/%");
|
|
|
|
}
|
2022-08-06 06:21:11 +02:00
|
|
|
}
|
2022-09-07 19:01:50 +02:00
|
|
|
for (final f in criteria.filters) {
|
|
|
|
f.apply(q);
|
|
|
|
}
|
|
|
|
return q.build();
|
|
|
|
});
|
|
|
|
// limit to supported formats only
|
|
|
|
query.where(db.files.contentType.like("image/%") |
|
|
|
|
db.files.contentType.like("video/%"));
|
|
|
|
for (final k in keywords) {
|
|
|
|
query.where(db.accountFiles.relativePath.like("%$k%"));
|
2022-08-06 06:21:11 +02:00
|
|
|
}
|
2022-09-07 19:01:50 +02:00
|
|
|
return await query
|
|
|
|
.map((r) => sql.CompleteFile(
|
|
|
|
r.readTable(db.files),
|
|
|
|
r.readTable(db.accountFiles),
|
|
|
|
r.readTableOrNull(db.images),
|
|
|
|
r.readTableOrNull(db.imageLocations),
|
|
|
|
r.readTableOrNull(db.trashes),
|
|
|
|
))
|
|
|
|
.get();
|
|
|
|
});
|
|
|
|
return await dbFiles.convertToAppFile(account);
|
|
|
|
} catch (e, stackTrace) {
|
|
|
|
_log.severe("[_listByPath] Failed while _listByPath", e, stackTrace);
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<List<File>> _listByLocation(
|
|
|
|
Account account, SearchCriteria criteria) async {
|
|
|
|
// location search requires exact match, for example, searching "united"
|
|
|
|
// will NOT return results from US, UK, UAE, etc. Searching by the alpha2
|
|
|
|
// code is supported
|
|
|
|
try {
|
|
|
|
final dbFiles = await _c.sqliteDb.use((db) async {
|
|
|
|
final query = db.queryFiles().run((q) {
|
|
|
|
q.setQueryMode(sql.FilesQueryMode.completeFile);
|
|
|
|
q.setAppAccount(account);
|
|
|
|
for (final r in account.roots) {
|
|
|
|
if (r.isNotEmpty) {
|
|
|
|
q.byOrRelativePathPattern("$r/%");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (final f in criteria.filters) {
|
|
|
|
f.apply(q);
|
|
|
|
}
|
|
|
|
q.byLocation(criteria.input);
|
|
|
|
return q.build();
|
|
|
|
});
|
|
|
|
// limit to supported formats only
|
|
|
|
query.where(db.files.contentType.like("image/%") |
|
|
|
|
db.files.contentType.like("video/%"));
|
|
|
|
return await query
|
|
|
|
.map((r) => sql.CompleteFile(
|
|
|
|
r.readTable(db.files),
|
|
|
|
r.readTable(db.accountFiles),
|
|
|
|
r.readTableOrNull(db.images),
|
|
|
|
r.readTableOrNull(db.imageLocations),
|
|
|
|
r.readTableOrNull(db.trashes),
|
|
|
|
))
|
|
|
|
.get();
|
|
|
|
});
|
|
|
|
return await dbFiles.convertToAppFile(account);
|
|
|
|
} catch (e, stackTrace) {
|
|
|
|
_log.severe(
|
|
|
|
"[_listByLocation] Failed while _listByLocation", e, stackTrace);
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<List<File>> _listByTag(
|
|
|
|
Account account, SearchCriteria criteria) async {
|
|
|
|
// tag search requires exact match, for example, searching "super" will NOT
|
|
|
|
// return results from "super tag"
|
|
|
|
try {
|
|
|
|
final dbTag = await _c.sqliteDb.use((db) async {
|
|
|
|
return await db.tagByDisplayName(
|
|
|
|
appAccount: account,
|
|
|
|
displayName: criteria.input,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
if (dbTag == null) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
final tag = SqliteTagConverter.fromSql(dbTag);
|
|
|
|
_log.info("[_listByTag] Found tag: ${tag.displayName}");
|
|
|
|
final files = await ListTaggedFile(_c)(account, [tag]);
|
|
|
|
return files
|
|
|
|
.where((f) => criteria.filters.every((c) => c.isSatisfy(f)))
|
|
|
|
.toList();
|
|
|
|
} catch (e, stackTrace) {
|
|
|
|
_log.severe("[_listByTag] Failed while _listByTag", e, stackTrace);
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<List<File>> _listByPerson(
|
|
|
|
Account account, SearchCriteria criteria) async {
|
|
|
|
// person search requires exact match of any parts, for example, searching
|
|
|
|
// "Ada" will return results from "Ada Crook" but NOT "Adabelle"
|
|
|
|
try {
|
|
|
|
final dbPersons = await _c.sqliteDb.use((db) async {
|
|
|
|
return await db.personsByName(
|
|
|
|
appAccount: account,
|
|
|
|
name: criteria.input,
|
|
|
|
);
|
2022-08-06 06:21:11 +02:00
|
|
|
});
|
2022-09-07 19:01:50 +02:00
|
|
|
if (dbPersons.isEmpty) {
|
|
|
|
return [];
|
2022-08-06 06:21:11 +02:00
|
|
|
}
|
2022-09-07 19:01:50 +02:00
|
|
|
final persons = await dbPersons.convertToAppPerson();
|
|
|
|
_log.info(
|
|
|
|
"[_listByPerson] Found people: ${persons.map((p) => p.name).toReadableString()}");
|
|
|
|
final futures = await Future.wait(
|
|
|
|
persons.map((p) async => await _c.faceRepo.list(account, p)));
|
|
|
|
final faces = futures.flatten().toList();
|
|
|
|
final files = await PopulatePerson(_c)(account, faces);
|
|
|
|
return files
|
|
|
|
.where((f) => criteria.filters.every((c) => c.isSatisfy(f)))
|
|
|
|
.toList();
|
|
|
|
} catch (e, stackTrace) {
|
|
|
|
_log.severe("[_listByPerson] Failed while _listByPerson", e, stackTrace);
|
|
|
|
return [];
|
|
|
|
}
|
2022-08-06 06:21:11 +02:00
|
|
|
}
|
|
|
|
|
2022-09-07 19:01:50 +02:00
|
|
|
final DiContainer _c;
|
2022-08-06 06:21:11 +02:00
|
|
|
}
|