2021-09-18 15:32:39 +02:00
package search
2020-05-08 15:41:01 +02:00
import (
"fmt"
2021-10-01 17:26:29 +02:00
"path"
2020-05-08 15:41:01 +02:00
"strings"
"time"
2022-03-30 20:36:25 +02:00
"github.com/photoprism/photoprism/internal/viewer"
2021-10-01 00:05:49 +02:00
"github.com/dustin/go-humanize/english"
2020-05-11 14:49:00 +02:00
"github.com/jinzhu/gorm"
2021-09-29 22:57:26 +02:00
2020-05-08 15:41:01 +02:00
"github.com/photoprism/photoprism/internal/entity"
"github.com/photoprism/photoprism/internal/form"
2022-03-30 20:36:25 +02:00
2021-10-01 17:26:29 +02:00
"github.com/photoprism/photoprism/pkg/fs"
2021-09-29 22:57:26 +02:00
"github.com/photoprism/photoprism/pkg/rnd"
2020-05-08 15:41:01 +02:00
"github.com/photoprism/photoprism/pkg/txt"
)
2022-03-30 20:36:25 +02:00
var PhotosColsAll = SelectString ( Photo { } , [ ] string { "*" } )
var PhotosColsView = SelectString ( Photo { } , SelectCols ( GeoResult { } , [ ] string { "*" } ) )
// Photos finds photos based on the search form provided and returns them as PhotoResults.
2021-11-26 13:59:10 +01:00
func Photos ( f form . SearchPhotos ) ( results PhotoResults , count int , err error ) {
2022-03-30 20:36:25 +02:00
return searchPhotos ( f , PhotosColsAll )
}
// PhotosViewerResults finds photos based on the search form provided and returns them as viewer.Results.
func PhotosViewerResults ( f form . SearchPhotos , contentUri , apiUri , previewToken , downloadToken string ) ( viewer . Results , int , error ) {
if results , count , err := searchPhotos ( f , PhotosColsView ) ; err != nil {
return viewer . Results { } , count , err
} else {
return results . ViewerResults ( contentUri , apiUri , previewToken , downloadToken ) , count , err
}
}
// photos searches for photos based on a Form and returns PhotoResults ([]Photo).
func searchPhotos ( f form . SearchPhotos , resultCols string ) ( results PhotoResults , count int , err error ) {
2020-05-23 20:58:58 +02:00
start := time . Now ( )
2022-03-30 20:36:25 +02:00
// Parse query string into fields.
2020-05-08 15:41:01 +02:00
if err := f . ParseQueryString ( ) ; err != nil {
2021-09-23 23:46:17 +02:00
return PhotoResults { } , 0 , err
2020-05-08 15:41:01 +02:00
}
s := UnscopedDb ( )
2021-09-20 12:36:59 +02:00
// s = s.LogMode(true)
2020-05-08 15:41:01 +02:00
2022-03-29 00:21:50 +02:00
// Database tables.
2022-03-30 20:36:25 +02:00
s = s . Table ( "files" ) . Select ( resultCols ) .
Joins ( "JOIN photos ON files.photo_id = photos.id AND files.media_id IS NOT NULL" ) .
2021-01-10 12:38:51 +01:00
Joins ( "LEFT JOIN cameras ON photos.camera_id = cameras.id" ) .
Joins ( "LEFT JOIN lenses ON photos.lens_id = lenses.id" ) .
2022-03-30 20:36:25 +02:00
Joins ( "LEFT JOIN places ON photos.place_id = places.id" )
2020-06-04 17:06:42 +02:00
2022-03-29 00:21:50 +02:00
// Offset and count.
2021-01-20 12:08:48 +01:00
if f . Count > 0 && f . Count <= MaxResults {
s = s . Limit ( f . Count ) . Offset ( f . Offset )
} else {
s = s . Limit ( MaxResults ) . Offset ( f . Offset )
}
2022-03-29 00:21:50 +02:00
// Sort order.
2021-01-20 12:08:48 +01:00
switch f . Order {
case entity . SortOrderEdited :
2022-03-30 20:36:25 +02:00
s = s . Where ( "photos.edited_at IS NOT NULL" ) . Order ( "photos.edited_at DESC, files.media_id" )
2021-01-20 12:08:48 +01:00
case entity . SortOrderRelevance :
if f . Label != "" {
2022-03-30 20:36:25 +02:00
s = s . Order ( "photos.photo_quality DESC, photos_labels.uncertainty ASC, files.time_index" )
2021-01-20 12:08:48 +01:00
} else {
2022-03-30 20:36:25 +02:00
s = s . Order ( "photos.photo_quality DESC, files.time_index" )
2021-01-20 12:08:48 +01:00
}
case entity . SortOrderNewest :
2022-03-30 20:36:25 +02:00
s = s . Order ( "files.time_index" )
2021-01-20 12:08:48 +01:00
case entity . SortOrderOldest :
2022-03-30 20:36:25 +02:00
s = s . Order ( "files.photo_taken_at, files.media_id" )
2021-01-20 12:08:48 +01:00
case entity . SortOrderSimilar :
s = s . Where ( "files.file_diff > 0" )
2022-03-30 20:36:25 +02:00
s = s . Order ( "photos.photo_color, photos.cell_id, files.file_diff, files.time_index" )
2021-01-20 12:08:48 +01:00
case entity . SortOrderName :
2022-03-30 20:36:25 +02:00
s = s . Order ( "photos.photo_path, photos.photo_name, files.time_index" )
2022-03-29 00:21:50 +02:00
case entity . SortOrderDefault , entity . SortOrderImported , entity . SortOrderAdded :
2022-03-30 20:36:25 +02:00
s = s . Order ( "files.media_id" )
2021-01-20 12:08:48 +01:00
default :
2022-03-29 00:21:50 +02:00
return PhotoResults { } , 0 , fmt . Errorf ( "invalid sort order" )
2021-01-20 12:08:48 +01:00
}
2022-03-29 00:21:50 +02:00
// Show hidden files?
2020-06-04 17:06:42 +02:00
if ! f . Hidden {
s = s . Where ( "files.file_type = 'jpg' OR files.file_video = 1" )
if f . Error {
s = s . Where ( "files.file_error <> ''" )
} else {
s = s . Where ( "files.file_error = ''" )
}
}
2020-05-08 15:41:01 +02:00
2022-03-29 00:21:50 +02:00
// Primary files only?
2020-06-28 15:23:15 +02:00
if f . Primary {
s = s . Where ( "files.file_primary = 1" )
}
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . UID ) {
2021-12-16 11:42:57 +01:00
s = s . Where ( "photos.photo_uid IN (?)" , strings . Split ( strings . ToLower ( f . UID ) , txt . Or ) )
2020-05-08 15:41:01 +02:00
2021-12-16 11:42:57 +01:00
// Take shortcut?
2021-12-16 15:26:54 +01:00
if f . Album == "" && f . Query == "" {
2022-03-30 20:36:25 +02:00
s = s . Order ( "files.media_id" )
2020-05-08 15:41:01 +02:00
2021-12-16 11:42:57 +01:00
if result := s . Scan ( & results ) ; result . Error != nil {
return results , 0 , result . Error
}
2020-05-23 20:58:58 +02:00
2021-12-16 15:26:54 +01:00
log . Debugf ( "photos: found %s for %s [%s]" , english . Plural ( len ( results ) , "result" , "results" ) , f . SerializeAll ( ) , time . Since ( start ) )
2020-05-08 15:41:01 +02:00
2021-12-16 11:42:57 +01:00
if f . Merged {
2022-03-29 00:21:50 +02:00
return results . Merge ( )
2021-12-16 11:42:57 +01:00
}
return results , len ( results ) , nil
}
2020-05-08 15:41:01 +02:00
}
2020-05-14 19:03:12 +02:00
// Filter by label, label category and keywords.
2020-05-08 15:41:01 +02:00
var categories [ ] entity . Category
2020-05-11 14:49:00 +02:00
var labels [ ] entity . Label
2020-05-08 15:41:01 +02:00
var labelIds [ ] uint
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Label ) {
2021-09-18 15:32:39 +02:00
if err := Db ( ) . Where ( AnySlug ( "label_slug" , f . Label , txt . Or ) ) . Or ( AnySlug ( "custom_slug" , f . Label , txt . Or ) ) . Find ( & labels ) . Error ; len ( labels ) == 0 || err != nil {
2021-12-14 17:07:38 +01:00
log . Debugf ( "search: label %s not found" , txt . LogParamLower ( f . Label ) )
2021-09-23 23:46:17 +02:00
return PhotoResults { } , 0 , nil
2020-05-08 15:41:01 +02:00
} else {
2020-05-29 18:04:30 +02:00
for _ , l := range labels {
labelIds = append ( labelIds , l . ID )
2022-03-30 20:36:25 +02:00
Log ( "find categories" , Db ( ) . Where ( "category_id = ?" , l . ID ) . Find ( & categories ) . Error )
log . Debugf ( "search: label %s includes %d categories" , txt . LogParamLower ( l . LabelName ) , len ( categories ) )
2020-05-08 15:41:01 +02:00
2020-05-29 18:04:30 +02:00
for _ , category := range categories {
labelIds = append ( labelIds , category . LabelID )
}
2020-05-08 15:41:01 +02:00
}
2022-03-29 00:21:50 +02:00
s = s . Joins ( "JOIN photos_labels ON photos_labels.photo_id = files.photo_id AND photos_labels.uncertainty < 100 AND photos_labels.label_id IN (?)" , labelIds ) .
2020-05-29 18:04:30 +02:00
Group ( "photos.id, files.id" )
2020-05-08 15:41:01 +02:00
}
}
2021-09-03 20:14:11 +02:00
// Set search filters based on search terms.
2021-09-17 15:52:25 +02:00
if terms := txt . SearchTerms ( f . Query ) ; f . Query != "" && len ( terms ) == 0 {
2021-09-29 22:57:26 +02:00
if f . Title == "" {
f . Title = fmt . Sprintf ( "%s*" , strings . Trim ( f . Query , "%*" ) )
2021-09-29 20:09:34 +02:00
f . Query = ""
}
2021-09-17 15:52:25 +02:00
} else if len ( terms ) > 0 {
2021-09-03 20:14:11 +02:00
switch {
case terms [ "faces" ] :
f . Query = strings . ReplaceAll ( f . Query , "faces" , "" )
f . Faces = "true"
2021-09-06 15:42:30 +02:00
case terms [ "people" ] :
f . Query = strings . ReplaceAll ( f . Query , "people" , "" )
f . Faces = "true"
2021-09-03 20:14:11 +02:00
case terms [ "videos" ] :
f . Query = strings . ReplaceAll ( f . Query , "videos" , "" )
f . Video = true
2021-10-12 14:31:27 +02:00
case terms [ "video" ] :
f . Query = strings . ReplaceAll ( f . Query , "video" , "" )
f . Video = true
2021-10-13 16:12:56 +02:00
case terms [ "live" ] :
f . Query = strings . ReplaceAll ( f . Query , "live" , "" )
f . Live = true
case terms [ "raws" ] :
f . Query = strings . ReplaceAll ( f . Query , "raws" , "" )
f . Raw = true
2021-09-03 20:14:11 +02:00
case terms [ "favorites" ] :
f . Query = strings . ReplaceAll ( f . Query , "favorites" , "" )
f . Favorite = true
case terms [ "stacks" ] :
f . Query = strings . ReplaceAll ( f . Query , "stacks" , "" )
f . Stack = true
case terms [ "panoramas" ] :
f . Query = strings . ReplaceAll ( f . Query , "panoramas" , "" )
f . Panorama = true
case terms [ "scans" ] :
f . Query = strings . ReplaceAll ( f . Query , "scans" , "" )
f . Scan = true
case terms [ "monochrome" ] :
f . Query = strings . ReplaceAll ( f . Query , "monochrome" , "" )
f . Mono = true
2021-10-12 14:31:27 +02:00
case terms [ "mono" ] :
f . Query = strings . ReplaceAll ( f . Query , "mono" , "" )
f . Mono = true
2021-09-03 20:14:11 +02:00
}
}
2021-08-30 11:26:57 +02:00
// Filter by location?
2020-07-11 23:43:29 +02:00
if f . Geo == true {
2020-07-12 08:27:05 +02:00
s = s . Where ( "photos.cell_id <> 'zz'" )
2020-05-08 15:41:01 +02:00
2021-08-29 19:19:54 +02:00
for _ , where := range LikeAnyKeyword ( "k.keyword" , f . Query ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_id IN (SELECT pk.photo_id FROM keywords k JOIN photos_keywords pk ON k.id = pk.keyword_id WHERE (?))" , gorm . Expr ( where ) )
2020-05-08 15:41:01 +02:00
}
} else if f . Query != "" {
2020-05-29 18:04:30 +02:00
if err := Db ( ) . Where ( AnySlug ( "custom_slug" , f . Query , " " ) ) . Find ( & labels ) . Error ; len ( labels ) == 0 || err != nil {
2021-12-14 17:07:38 +01:00
log . Debugf ( "search: label %s not found, using fuzzy search" , txt . LogParamLower ( f . Query ) )
2020-05-08 15:41:01 +02:00
2021-08-29 19:19:54 +02:00
for _ , where := range LikeAnyKeyword ( "k.keyword" , f . Query ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_id IN (SELECT pk.photo_id FROM keywords k JOIN photos_keywords pk ON k.id = pk.keyword_id WHERE (?))" , gorm . Expr ( where ) )
2020-05-11 14:49:00 +02:00
}
2020-05-08 15:41:01 +02:00
} else {
2020-05-11 14:49:00 +02:00
for _ , l := range labels {
labelIds = append ( labelIds , l . ID )
2020-05-08 15:41:01 +02:00
2020-05-11 14:49:00 +02:00
Db ( ) . Where ( "category_id = ?" , l . ID ) . Find ( & categories )
2020-05-08 15:41:01 +02:00
2021-12-14 17:07:38 +01:00
log . Debugf ( "search: label %s includes %d categories" , txt . LogParamLower ( l . LabelName ) , len ( categories ) )
2020-05-08 15:41:01 +02:00
2020-05-11 14:49:00 +02:00
for _ , category := range categories {
labelIds = append ( labelIds , category . LabelID )
}
}
2020-05-08 15:41:01 +02:00
2021-08-29 19:19:54 +02:00
if wheres := LikeAnyKeyword ( "k.keyword" , f . Query ) ; len ( wheres ) > 0 {
2021-08-29 16:16:49 +02:00
for _ , where := range wheres {
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_id IN (SELECT pk.photo_id FROM keywords k JOIN photos_keywords pk ON k.id = pk.keyword_id WHERE (?)) OR " +
"files.photo_id IN (SELECT pl.photo_id FROM photos_labels pl WHERE pl.uncertainty < 100 AND pl.label_id IN (?))" , gorm . Expr ( where ) , labelIds )
2021-08-29 16:16:49 +02:00
}
2020-05-11 14:49:00 +02:00
} else {
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_id IN (SELECT pl.photo_id FROM photos_labels pl WHERE pl.uncertainty < 100 AND pl.label_id IN (?))" , labelIds )
2020-05-11 14:49:00 +02:00
}
2020-05-08 15:41:01 +02:00
}
}
2021-08-29 16:16:49 +02:00
// Search for one or more keywords?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Keywords ) {
2022-01-05 18:51:18 +01:00
for _ , where := range LikeAnyWord ( "k.keyword" , f . Keywords ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_id IN (SELECT pk.photo_id FROM keywords k JOIN photos_keywords pk ON k.id = pk.keyword_id WHERE (?))" , gorm . Expr ( where ) )
2021-08-29 16:16:49 +02:00
}
}
2021-10-13 16:12:56 +02:00
// Filter by number of faces?
2021-10-12 14:31:27 +02:00
if txt . IsUInt ( f . Faces ) {
s = s . Where ( "photos.photo_faces >= ?" , txt . Int ( f . Faces ) )
2021-10-13 16:12:56 +02:00
} else if txt . New ( f . Faces ) && f . Face == "" {
f . Face = f . Faces
f . Faces = ""
2021-10-12 14:31:27 +02:00
} else if txt . Yes ( f . Faces ) {
s = s . Where ( "photos.photo_faces > 0" )
} else if txt . No ( f . Faces ) {
s = s . Where ( "photos.photo_faces = 0" )
}
2021-09-23 14:23:00 +02:00
// Filter for specific face clusters? Example: PLJ7A3G4MBGZJRMVDIUCBLC46IAP4N7O
if len ( f . Face ) >= 32 {
2021-09-18 15:32:39 +02:00
for _ , f := range strings . Split ( strings . ToUpper ( f . Face ) , txt . And ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( fmt . Sprintf ( "files.photo_id IN (SELECT photo_id FROM files f JOIN %s m ON f.file_uid = m.file_uid AND m.marker_invalid = 0 WHERE face_id IN (?))" ,
2021-09-18 15:32:39 +02:00
entity . Marker { } . TableName ( ) ) , strings . Split ( f , txt . Or ) )
}
2021-09-29 20:09:34 +02:00
} else if txt . New ( f . Face ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( fmt . Sprintf ( "files.photo_id IN (SELECT photo_id FROM files f JOIN %s m ON f.file_uid = m.file_uid AND m.marker_invalid = 0 AND m.marker_type = ? WHERE subj_uid IS NULL OR subj_uid = '')" ,
2021-09-29 20:09:34 +02:00
entity . Marker { } . TableName ( ) ) , entity . MarkerFace )
2021-09-23 14:23:00 +02:00
} else if txt . No ( f . Face ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( fmt . Sprintf ( "files.photo_id IN (SELECT photo_id FROM files f JOIN %s m ON f.file_uid = m.file_uid AND m.marker_invalid = 0 AND m.marker_type = ? WHERE face_id IS NULL OR face_id = '')" ,
2021-09-23 14:23:00 +02:00
entity . Marker { } . TableName ( ) ) , entity . MarkerFace )
} else if txt . Yes ( f . Face ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( fmt . Sprintf ( "files.photo_id IN (SELECT photo_id FROM files f JOIN %s m ON f.file_uid = m.file_uid AND m.marker_invalid = 0 AND m.marker_type = ? WHERE face_id IS NOT NULL AND face_id <> '')" ,
2021-09-23 14:23:00 +02:00
entity . Marker { } . TableName ( ) ) , entity . MarkerFace )
2021-09-18 15:32:39 +02:00
}
2021-08-29 16:16:49 +02:00
// Filter for one or more subjects?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Subject ) {
2021-09-18 15:32:39 +02:00
for _ , subj := range strings . Split ( strings . ToLower ( f . Subject ) , txt . And ) {
2021-09-20 12:36:59 +02:00
if subjects := strings . Split ( subj , txt . Or ) ; rnd . ContainsUIDs ( subjects , 'j' ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( fmt . Sprintf ( "files.photo_id IN (SELECT photo_id FROM files f JOIN %s m ON f.file_uid = m.file_uid AND m.marker_invalid = 0 WHERE subj_uid IN (?))" ,
2021-09-20 12:36:59 +02:00
entity . Marker { } . TableName ( ) ) , subjects )
} else {
2022-03-29 00:21:50 +02:00
s = s . Where ( fmt . Sprintf ( "files.photo_id IN (SELECT photo_id FROM files f JOIN %s m ON f.file_uid = m.file_uid AND m.marker_invalid = 0 JOIN %s s ON s.subj_uid = m.subj_uid WHERE (?))" ,
2021-09-20 12:36:59 +02:00
entity . Marker { } . TableName ( ) , entity . Subject { } . TableName ( ) ) , gorm . Expr ( AnySlug ( "s.subj_slug" , subj , txt . Or ) ) )
}
2021-08-30 11:56:34 +02:00
}
2022-03-24 18:30:59 +01:00
} else if txt . NotEmpty ( f . Subjects ) {
2021-09-20 09:57:48 +02:00
for _ , where := range LikeAllNames ( Cols { "subj_name" , "subj_alias" } , f . Subjects ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( fmt . Sprintf ( "files.photo_id IN (SELECT photo_id FROM files f JOIN %s m ON f.file_uid = m.file_uid AND m.marker_invalid = 0 JOIN %s s ON s.subj_uid = m.subj_uid WHERE (?))" ,
2021-08-29 16:16:49 +02:00
entity . Marker { } . TableName ( ) , entity . Subject { } . TableName ( ) ) , gorm . Expr ( where ) )
}
}
// Filter by status?
2020-06-04 14:56:27 +02:00
if f . Hidden {
s = s . Where ( "photos.photo_quality = -1" )
s = s . Where ( "photos.deleted_at IS NULL" )
} else if f . Archived {
2020-12-15 20:14:06 +01:00
s = s . Where ( "photos.photo_quality > -1" )
2020-05-08 15:41:01 +02:00
s = s . Where ( "photos.deleted_at IS NOT NULL" )
} else {
s = s . Where ( "photos.deleted_at IS NULL" )
if f . Private {
s = s . Where ( "photos.photo_private = 1" )
} else if f . Public {
s = s . Where ( "photos.photo_private = 0" )
}
if f . Review {
s = s . Where ( "photos.photo_quality < 3" )
} else if f . Quality != 0 && f . Private == false {
s = s . Where ( "photos.photo_quality >= ?" , f . Quality )
}
}
2022-03-24 18:30:59 +01:00
// Filter by camera id or name?
if txt . IsPosInt ( f . Camera ) {
s = s . Where ( "photos.camera_id = ?" , txt . UInt ( f . Camera ) )
} else if txt . NotEmpty ( f . Camera ) {
v := strings . Trim ( f . Camera , "*%" ) + "%"
2022-03-25 18:01:34 +01:00
s = s . Where ( "cameras.camera_name LIKE ? OR cameras.camera_model LIKE ? OR cameras.camera_slug LIKE ?" , v , v , v )
2020-05-08 15:41:01 +02:00
}
2022-03-24 18:30:59 +01:00
// Filter by lens id or name?
if txt . IsPosInt ( f . Lens ) {
s = s . Where ( "photos.lens_id = ?" , txt . UInt ( f . Lens ) )
} else if txt . NotEmpty ( f . Lens ) {
v := strings . Trim ( f . Lens , "*%" ) + "%"
2022-03-25 18:01:34 +01:00
s = s . Where ( "lenses.lens_name LIKE ? OR lenses.lens_model LIKE ? OR lenses.lens_slug LIKE ?" , v , v , v )
2020-05-08 15:41:01 +02:00
}
2021-08-29 16:16:49 +02:00
// Filter by year?
2021-09-20 23:32:35 +02:00
if f . Year != "" {
s = s . Where ( AnyInt ( "photos.photo_year" , f . Year , txt . Or , entity . UnknownYear , txt . YearMax ) )
2020-05-08 15:41:01 +02:00
}
2021-08-29 16:16:49 +02:00
// Filter by month?
2021-09-20 23:32:35 +02:00
if f . Month != "" {
s = s . Where ( AnyInt ( "photos.photo_month" , f . Month , txt . Or , entity . UnknownMonth , txt . MonthMax ) )
2020-05-08 15:41:01 +02:00
}
2021-08-29 16:16:49 +02:00
// Filter by day?
2021-09-20 23:32:35 +02:00
if f . Day != "" {
s = s . Where ( AnyInt ( "photos.photo_day" , f . Day , txt . Or , entity . UnknownDay , txt . DayMax ) )
2020-07-06 07:41:33 +02:00
}
2021-10-13 16:12:56 +02:00
// Filter by main color?
2020-05-08 15:41:01 +02:00
if f . Color != "" {
2021-09-18 15:32:39 +02:00
s = s . Where ( "files.file_main_color IN (?)" , strings . Split ( strings . ToLower ( f . Color ) , txt . Or ) )
2020-05-08 15:41:01 +02:00
}
2021-10-13 16:12:56 +02:00
// Find favorites only?
2020-05-14 19:03:12 +02:00
if f . Favorite {
2020-05-08 15:41:01 +02:00
s = s . Where ( "photos.photo_favorite = 1" )
}
2021-10-13 16:12:56 +02:00
// Find scans only?
2020-07-06 14:35:25 +02:00
if f . Scan {
s = s . Where ( "photos.photo_scan = 1" )
2020-07-05 14:48:49 +02:00
}
2021-10-13 16:12:56 +02:00
// Find panoramas only?
2020-07-16 13:02:48 +02:00
if f . Panorama {
s = s . Where ( "photos.photo_panorama = 1" )
}
2021-10-13 16:12:56 +02:00
// Find portraits only?
if f . Portrait {
s = s . Where ( "files.file_portrait = 1" )
}
2020-12-19 19:15:32 +01:00
if f . Stackable {
s = s . Where ( "photos.photo_stack > -1" )
} else if f . Unstacked {
s = s . Where ( "photos.photo_stack = -1" )
2020-12-11 17:21:13 +01:00
}
2021-10-13 16:12:56 +02:00
// Filter by location country?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Country ) {
2021-09-18 15:32:39 +02:00
s = s . Where ( "photos.photo_country IN (?)" , strings . Split ( strings . ToLower ( f . Country ) , txt . Or ) )
2020-05-08 15:41:01 +02:00
}
2021-10-13 16:12:56 +02:00
// Filter by location state?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . State ) {
2021-09-18 15:32:39 +02:00
s = s . Where ( "places.place_state IN (?)" , strings . Split ( f . State , txt . Or ) )
2020-05-29 18:04:30 +02:00
}
2021-10-13 16:12:56 +02:00
// Filter by location category?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Category ) {
2020-07-12 08:27:05 +02:00
s = s . Joins ( "JOIN cells ON photos.cell_id = cells.id" ) .
2021-09-18 15:32:39 +02:00
Where ( "cells.cell_category IN (?)" , strings . Split ( strings . ToLower ( f . Category ) , txt . Or ) )
2020-05-29 18:04:30 +02:00
}
2021-10-13 16:12:56 +02:00
// Filter by media type?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Type ) {
2021-09-18 15:32:39 +02:00
s = s . Where ( "photos.photo_type IN (?)" , strings . Split ( strings . ToLower ( f . Type ) , txt . Or ) )
2021-10-13 16:12:56 +02:00
} else if f . Video {
2021-11-30 15:38:24 +01:00
s = s . Where ( "photos.photo_type = 'video'" )
2020-05-21 13:26:28 +02:00
} else if f . Photo {
s = s . Where ( "photos.photo_type IN ('image','raw','live')" )
2021-10-13 16:12:56 +02:00
} else if f . Raw {
s = s . Where ( "photos.photo_type = 'raw'" )
} else if f . Live {
s = s . Where ( "photos.photo_type = 'live'" )
2020-05-21 10:03:56 +02:00
}
2021-10-13 16:12:56 +02:00
// Filter by storage path?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Path ) {
2020-05-21 10:03:56 +02:00
p := f . Path
if strings . HasPrefix ( p , "/" ) {
p = p [ 1 : ]
}
if strings . HasSuffix ( p , "/" ) {
s = s . Where ( "photos.photo_path = ?" , p [ : len ( p ) - 1 ] )
} else {
2021-09-29 20:09:34 +02:00
where , values := OrLike ( "photos.photo_path" , p )
s = s . Where ( where , values ... )
2020-05-21 10:03:56 +02:00
}
}
2021-10-01 17:26:29 +02:00
// Filter by primary file name without path and extension.
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Name ) {
2021-10-01 17:26:29 +02:00
where , names := OrLike ( "photos.photo_name" , f . Name )
// Omit file path and known extensions.
for i := range names {
names [ i ] = fs . StripKnownExt ( path . Base ( names [ i ] . ( string ) ) )
}
s = s . Where ( where , names ... )
2020-05-21 10:03:56 +02:00
}
2021-10-13 16:12:56 +02:00
// Filter by complete file names?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Filename ) {
2021-09-29 20:09:34 +02:00
where , values := OrLike ( "files.file_name" , f . Filename )
s = s . Where ( where , values ... )
2020-12-03 21:07:38 +01:00
}
2021-10-13 16:12:56 +02:00
// Filter by original file name?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Original ) {
2021-09-29 20:09:34 +02:00
where , values := OrLike ( "photos.original_name" , f . Original )
s = s . Where ( where , values ... )
2020-06-03 10:33:09 +02:00
}
2021-10-13 16:12:56 +02:00
// Filter by photo title?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Title ) {
2021-09-29 20:09:34 +02:00
where , values := OrLike ( "photos.photo_title" , f . Title )
s = s . Where ( where , values ... )
2020-05-08 15:41:01 +02:00
}
2021-10-13 16:12:56 +02:00
// Filter by file hash?
2022-03-24 18:30:59 +01:00
if txt . NotEmpty ( f . Hash ) {
2021-09-18 15:32:39 +02:00
s = s . Where ( "files.file_hash IN (?)" , strings . Split ( strings . ToLower ( f . Hash ) , txt . Or ) )
2020-05-08 15:41:01 +02:00
}
if f . Mono {
2020-12-19 01:56:00 +01:00
s = s . Where ( "files.file_chroma = 0 OR file_colors = '111111111'" )
2020-05-08 15:41:01 +02:00
} else if f . Chroma > 9 {
s = s . Where ( "files.file_chroma > ?" , f . Chroma )
} else if f . Chroma > 0 {
s = s . Where ( "files.file_chroma > 0 AND files.file_chroma <= ?" , f . Chroma )
}
if f . Diff != 0 {
s = s . Where ( "files.file_diff = ?" , f . Diff )
}
if f . Fmin > 0 {
s = s . Where ( "photos.photo_f_number >= ?" , f . Fmin )
}
if f . Fmax > 0 {
s = s . Where ( "photos.photo_f_number <= ?" , f . Fmax )
}
if f . Dist == 0 {
f . Dist = 20
} else if f . Dist > 5000 {
f . Dist = 5000
}
2022-03-30 20:36:25 +02:00
// Filter by approx distance to co-ordinates:
2021-02-11 20:06:23 +01:00
if f . Lat != 0 {
2021-09-18 15:32:39 +02:00
latMin := f . Lat - Radius * float32 ( f . Dist )
latMax := f . Lat + Radius * float32 ( f . Dist )
2020-05-08 15:41:01 +02:00
s = s . Where ( "photos.photo_lat BETWEEN ? AND ?" , latMin , latMax )
}
2021-02-11 20:06:23 +01:00
if f . Lng != 0 {
2021-09-18 15:32:39 +02:00
lngMin := f . Lng - Radius * float32 ( f . Dist )
lngMax := f . Lng + Radius * float32 ( f . Dist )
2020-05-08 15:41:01 +02:00
s = s . Where ( "photos.photo_lng BETWEEN ? AND ?" , lngMin , lngMax )
}
if ! f . Before . IsZero ( ) {
s = s . Where ( "photos.taken_at <= ?" , f . Before . Format ( "2006-01-02" ) )
}
if ! f . After . IsZero ( ) {
s = s . Where ( "photos.taken_at >= ?" , f . After . Format ( "2006-01-02" ) )
}
2021-08-29 16:16:49 +02:00
// Find stacks only?
2020-07-13 15:59:54 +02:00
if f . Stack {
2022-03-30 20:36:25 +02:00
s = s . Where ( "photos.id IN (SELECT a.photo_id FROM files a JOIN files b ON a.id != b.id AND a.photo_id = b.photo_id AND a.file_type = b.file_type WHERE a.file_type='jpg')" )
2020-07-05 17:22:26 +02:00
}
2021-08-29 16:16:49 +02:00
// Filter by album?
2021-09-20 22:58:11 +02:00
if rnd . IsPPID ( f . Album , 'a' ) {
2020-05-30 01:41:47 +02:00
if f . Filter != "" {
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_uid NOT IN (SELECT photo_uid FROM photos_albums pa WHERE pa.hidden = 1 AND pa.album_uid = ?)" , f . Album )
2020-05-30 01:41:47 +02:00
} else {
2022-03-29 00:21:50 +02:00
s = s . Joins ( "JOIN photos_albums ON photos_albums.photo_uid = files.photo_uid" ) .
2021-09-20 22:58:11 +02:00
Where ( "photos_albums.hidden = 0 AND photos_albums.album_uid = ?" , f . Album )
2020-05-30 01:41:47 +02:00
}
2020-07-05 14:48:49 +02:00
} else if f . Unsorted && f . Filter == "" {
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_uid NOT IN (SELECT photo_uid FROM photos_albums pa WHERE pa.hidden = 0)" )
2022-03-24 18:30:59 +01:00
} else if txt . NotEmpty ( f . Album ) {
v := strings . Trim ( f . Album , "*%" ) + "%"
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_uid IN (SELECT pa.photo_uid FROM photos_albums pa JOIN albums a ON a.album_uid = pa.album_uid AND pa.hidden = 0 WHERE (a.album_title LIKE ? OR a.album_slug LIKE ?))" , v , v )
2022-03-24 18:30:59 +01:00
} else if txt . NotEmpty ( f . Albums ) {
2021-08-29 19:19:54 +02:00
for _ , where := range LikeAnyWord ( "a.album_title" , f . Albums ) {
2022-03-29 00:21:50 +02:00
s = s . Where ( "files.photo_uid IN (SELECT pa.photo_uid FROM photos_albums pa JOIN albums a ON a.album_uid = pa.album_uid AND pa.hidden = 0 WHERE (?))" , gorm . Expr ( where ) )
2021-08-29 16:16:49 +02:00
}
2020-05-30 01:41:47 +02:00
}
2020-05-30 14:52:47 +02:00
if err := s . Scan ( & results ) . Error ; err != nil {
return results , 0 , err
2020-05-08 15:41:01 +02:00
}
2021-12-16 15:26:54 +01:00
log . Debugf ( "photos: found %s for %s [%s]" , english . Plural ( len ( results ) , "result" , "results" ) , f . SerializeAll ( ) , time . Since ( start ) )
2020-05-23 20:58:58 +02:00
2020-05-08 15:41:01 +02:00
if f . Merged {
2022-03-29 00:21:50 +02:00
return results . Merge ( )
2020-05-08 15:41:01 +02:00
}
return results , len ( results ) , nil
}