b6378a5c1f
Signed-off-by: Michael Mayer <michael@photoprism.app>
320 lines
9.1 KiB
Go
320 lines
9.1 KiB
Go
package query
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/photoprism/photoprism/internal/entity"
|
|
"github.com/photoprism/photoprism/internal/maps"
|
|
"github.com/photoprism/photoprism/pkg/clean"
|
|
"github.com/photoprism/photoprism/pkg/txt"
|
|
)
|
|
|
|
// Moment contains photo counts per month and year
|
|
type Moment struct {
|
|
Label string `json:"Label"`
|
|
Year int `json:"Year"`
|
|
Month int `json:"Month"`
|
|
State string `json:"State"`
|
|
Country string `json:"Country"`
|
|
PhotoCount int `json:"PhotoCount"`
|
|
}
|
|
|
|
var MomentLabels = map[string]string{
|
|
"park": "Parks & Gardens",
|
|
"botanical-garden": "Parks & Gardens",
|
|
"water-park": "Parks & Gardens",
|
|
"alpine": "Outdoor Adventures",
|
|
"hiking": "Outdoor Adventures",
|
|
"mountain": "Outdoor Adventures",
|
|
"mountains": "Outdoor Adventures",
|
|
"camping": "Outdoor Adventures",
|
|
"camper": "Outdoor Adventures",
|
|
"bench": "Outdoor Adventures",
|
|
"bunker": "Outdoor Adventures",
|
|
"castle": "Outdoor Adventures",
|
|
"viewpoint": "Outdoor Adventures",
|
|
"nature-reserve": "Nature & Landscape",
|
|
"landscape": "Nature & Landscape",
|
|
"nature": "Nature & Landscape",
|
|
"flower": "Nature & Landscape",
|
|
"field": "Nature & Landscape",
|
|
"forest": "Nature & Landscape",
|
|
"rocks": "Nature & Landscape",
|
|
"valley": "Nature & Landscape",
|
|
"bay": "Bays, Capes & Beaches",
|
|
"beach": "Bays, Capes & Beaches",
|
|
"seashore": "Bays, Capes & Beaches",
|
|
"cape": "Bays, Capes & Beaches",
|
|
"ship": "Water, Ships & Boats",
|
|
"water": "Water, Ships & Boats",
|
|
"pier": "Water, Ships & Boats",
|
|
"boat": "Water, Ships & Boats",
|
|
"boathouse": "Water, Ships & Boats",
|
|
"lakeside": "Water, Ships & Boats",
|
|
"shark": "Water, Ships & Boats",
|
|
"fish": "Water, Ships & Boats",
|
|
"jellyfish": "Water, Ships & Boats",
|
|
"submarine": "Water, Ships & Boats",
|
|
"diving": "Water, Ships & Boats",
|
|
"festival": "Festivals & Entertainment",
|
|
"nightclub": "Festivals & Entertainment",
|
|
"microphone": "Festivals & Entertainment",
|
|
"stage": "Festivals & Entertainment",
|
|
"theater": "Festivals & Entertainment",
|
|
"theme park": "Festivals & Entertainment",
|
|
"event": "Festivals & Entertainment",
|
|
"wine": "Festivals & Entertainment",
|
|
"cat": "Pets",
|
|
"dog": "Pets",
|
|
"rabbit": "Pets",
|
|
"hamster": "Pets",
|
|
}
|
|
|
|
// MomentLabelsFilter returns the smart filter string for a moment based on a matching label.
|
|
func MomentLabelsFilter(label string) string {
|
|
// TODO: Needs refactoring
|
|
label = strings.SplitN(label, txt.Or, 2)[0]
|
|
|
|
title := MomentLabels[label]
|
|
|
|
if title == "" {
|
|
return ""
|
|
}
|
|
|
|
var l []string
|
|
|
|
for i := range MomentLabels {
|
|
if MomentLabels[i] == title {
|
|
l = append(l, i)
|
|
}
|
|
}
|
|
|
|
return strings.Join(txt.UniqueWords(l), txt.Or)
|
|
}
|
|
|
|
// CountryName returns the country name if any.
|
|
func (m Moment) CountryName() string {
|
|
if m.Country == "" {
|
|
return ""
|
|
}
|
|
|
|
return maps.CountryName(m.Country)
|
|
}
|
|
|
|
// Location returns the location name for the moment.
|
|
func (m Moment) Location() string {
|
|
if state := clean.State(m.State, m.Country); state != "" {
|
|
return fmt.Sprintf("%s, %s", state, m.CountryName())
|
|
}
|
|
|
|
return m.CountryName()
|
|
}
|
|
|
|
// Slug returns an identifier string for a moment.
|
|
func (m Moment) Slug() (s string) {
|
|
state := clean.State(m.State, m.Country)
|
|
|
|
if state == "" {
|
|
return m.TitleSlug()
|
|
}
|
|
|
|
country := maps.CountryName(m.Country)
|
|
|
|
if m.Year > 1900 && m.Month == 0 {
|
|
s = fmt.Sprintf("%s-%s-%04d", country, state, m.Year)
|
|
} else if m.Year > 1900 && m.Month > 0 && m.Month <= 12 {
|
|
s = fmt.Sprintf("%s-%s-%04d-%02d", country, state, m.Year, m.Month)
|
|
} else {
|
|
s = fmt.Sprintf("%s-%s", country, state)
|
|
}
|
|
|
|
return txt.Slug(s)
|
|
}
|
|
|
|
// TitleSlug returns an identifier string based on the title.
|
|
func (m Moment) TitleSlug() string {
|
|
return txt.Slug(m.Title())
|
|
}
|
|
|
|
// Title returns an english title for the moment.
|
|
func (m Moment) Title() string {
|
|
state := clean.State(m.State, m.Country)
|
|
|
|
if m.Year == 0 && m.Month == 0 {
|
|
if m.Label != "" {
|
|
return MomentLabels[strings.SplitN(m.Label, txt.Or, 2)[0]]
|
|
}
|
|
if state != "" {
|
|
return state
|
|
}
|
|
|
|
return m.CountryName()
|
|
}
|
|
|
|
if m.Country != "" && m.Year > 1900 && m.Month == 0 {
|
|
if state != "" {
|
|
return fmt.Sprintf("%s / %d", state, m.Year)
|
|
}
|
|
|
|
return fmt.Sprintf("%s %d", m.CountryName(), m.Year)
|
|
}
|
|
|
|
if m.Year > 1900 && m.Month > 0 && m.Month <= 12 {
|
|
date := time.Date(m.Year, time.Month(m.Month), 1, 0, 0, 0, 0, time.UTC)
|
|
|
|
if state != "" {
|
|
return fmt.Sprintf("%s / %s", state, date.Format("January 2006"))
|
|
}
|
|
|
|
if m.Country == "" {
|
|
return date.Format("January 2006")
|
|
}
|
|
|
|
return fmt.Sprintf("%s / %s", m.CountryName(), date.Format("January 2006"))
|
|
}
|
|
|
|
if m.Month > 0 && m.Month <= 12 {
|
|
return time.Month(m.Month).String()
|
|
}
|
|
|
|
return m.CountryName()
|
|
}
|
|
|
|
// Moments represents a list of moments.
|
|
type Moments []Moment
|
|
|
|
// MomentsTime counts photos by month and year.
|
|
func MomentsTime(threshold int, public bool) (results Moments, err error) {
|
|
stmt := UnscopedDb().Table("photos").
|
|
Select("photos.photo_year AS year, photos.photo_month AS month, COUNT(*) AS photo_count").
|
|
Where("photos.photo_quality >= 3 AND deleted_at IS NULL AND photos.photo_year > 0 AND photos.photo_month > 0")
|
|
|
|
// Ignore private pictures?
|
|
if public {
|
|
stmt = stmt.Where("photo_private = 0")
|
|
}
|
|
|
|
stmt = stmt.Group("photos.photo_year, photos.photo_month").
|
|
Order("photos.photo_year DESC, photos.photo_month DESC").
|
|
Having("photo_count >= ?", threshold)
|
|
|
|
if err = stmt.Scan(&results).Error; err != nil {
|
|
return results, err
|
|
}
|
|
|
|
return results, nil
|
|
}
|
|
|
|
// MomentsCountries returns the most popular countries by year.
|
|
func MomentsCountries(threshold int, public bool) (results Moments, err error) {
|
|
stmt := UnscopedDb().Table("photos").
|
|
Select("photo_year AS year, photo_country AS country, COUNT(*) AS photo_count").
|
|
Where("photos.photo_quality >= 3 AND deleted_at IS NULL AND photo_country <> 'zz' AND photo_year > 0")
|
|
|
|
// Ignore private pictures?
|
|
if public {
|
|
stmt = stmt.Where("photo_private = 0")
|
|
}
|
|
|
|
stmt = stmt.Group("photo_year, photo_country").
|
|
Having("photo_count >= ?", threshold)
|
|
|
|
if err = stmt.Scan(&results).Error; err != nil {
|
|
return results, err
|
|
}
|
|
|
|
return results, nil
|
|
}
|
|
|
|
// MomentsStates returns the most popular states and countries by year.
|
|
func MomentsStates(threshold int, public bool) (results Moments, err error) {
|
|
stmt := UnscopedDb().Table("photos").
|
|
Select("p.place_country AS country, p.place_state AS state, COUNT(*) AS photo_count").
|
|
Joins("JOIN places p ON p.id = photos.place_id").
|
|
Where("photos.photo_quality >= 3 AND photos.deleted_at IS NULL AND p.place_state <> '' AND p.place_country <> 'zz'")
|
|
|
|
// Ignore private pictures?
|
|
if public {
|
|
stmt = stmt.Where("photo_private = 0")
|
|
}
|
|
|
|
stmt = stmt.Group("p.place_country, p.place_state").
|
|
Having("photo_count >= ?", threshold)
|
|
|
|
if err = stmt.Scan(&results).Error; err != nil {
|
|
return results, err
|
|
}
|
|
|
|
return results, nil
|
|
}
|
|
|
|
// MomentsLabels returns the most popular photo labels.
|
|
func MomentsLabels(threshold int, public bool) (results Moments, err error) {
|
|
var cats []string
|
|
|
|
for cat := range MomentLabels {
|
|
cats = append(cats, cat)
|
|
}
|
|
|
|
m := Moments{}
|
|
|
|
stmt := UnscopedDb().Table("photos").
|
|
Select("l.label_slug AS label, COUNT(*) AS photo_count").
|
|
Joins("JOIN photos_labels pl ON pl.photo_id = photos.id AND pl.uncertainty < 100").
|
|
Joins("JOIN labels l ON l.id = pl.label_id").
|
|
Where("photos.photo_quality >= 3 AND photos.deleted_at IS NULL AND l.label_slug IN (?)", cats)
|
|
|
|
// Ignore private pictures?
|
|
if public {
|
|
stmt = stmt.Where("photo_private = 0")
|
|
}
|
|
|
|
stmt = stmt.Group("l.label_slug").
|
|
Having("photo_count >= ?", threshold)
|
|
|
|
if err = stmt.Scan(&m).Error; err != nil {
|
|
return m, err
|
|
}
|
|
|
|
done := make(map[string]bool)
|
|
|
|
for i := 0; i < len(m); i++ {
|
|
f := MomentLabelsFilter(m[i].Label)
|
|
|
|
if _, ok := done[f]; ok {
|
|
continue
|
|
} else {
|
|
done[f] = true
|
|
}
|
|
|
|
m[i].Label = f
|
|
results = append(results, m[i])
|
|
}
|
|
|
|
return results, nil
|
|
}
|
|
|
|
// RemoveDuplicateMoments deletes generated albums with duplicate slug or filter.
|
|
func RemoveDuplicateMoments() (removed int, err error) {
|
|
if res := UnscopedDb().Exec(`DELETE FROM links WHERE share_uid
|
|
IN (SELECT a.album_uid FROM albums a JOIN albums b ON a.album_type <> ?
|
|
AND a.album_type = b.album_type AND a.id > b.id
|
|
WHERE (a.album_slug = b.album_slug OR a.album_filter = b.album_filter)
|
|
GROUP BY a.album_uid)`, entity.AlbumManual); res.Error != nil {
|
|
return removed, res.Error
|
|
}
|
|
|
|
if res := UnscopedDb().Exec(`DELETE FROM albums WHERE id
|
|
IN (SELECT a.id FROM albums a JOIN albums b ON a.album_type <> ?
|
|
AND a.album_type = b.album_type AND a.id > b.id
|
|
WHERE (a.album_slug = b.album_slug OR a.album_filter = b.album_filter)
|
|
GROUP BY a.album_uid)`, entity.AlbumManual); res.Error != nil {
|
|
return removed, res.Error
|
|
} else if res.RowsAffected > 0 {
|
|
removed = int(res.RowsAffected)
|
|
}
|
|
|
|
return removed, nil
|
|
}
|