2019-12-28 20:24:20 +01:00
|
|
|
package entity
|
|
|
|
|
|
|
|
import (
|
2020-05-26 21:51:34 +02:00
|
|
|
"strings"
|
2020-12-13 15:43:01 +01:00
|
|
|
"sync"
|
2019-12-28 20:24:20 +01:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/photoprism/photoprism/internal/maps"
|
2022-04-15 09:42:07 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/clean"
|
2019-12-28 20:24:20 +01:00
|
|
|
)
|
|
|
|
|
2020-12-13 15:43:01 +01:00
|
|
|
var placeMutex = sync.Mutex{}
|
|
|
|
|
2021-11-20 16:36:34 +01:00
|
|
|
// Place represents a distinct region identified by city, district, state, and country.
|
2019-12-28 20:24:20 +01:00
|
|
|
type Place struct {
|
2020-09-13 17:51:43 +02:00
|
|
|
ID string `gorm:"type:VARBINARY(42);primary_key;auto_increment:false;" json:"PlaceID" yaml:"PlaceID"`
|
2021-11-20 16:36:34 +01:00
|
|
|
PlaceLabel string `gorm:"type:VARCHAR(400);" json:"Label" yaml:"Label"`
|
2021-11-18 00:46:34 +01:00
|
|
|
PlaceDistrict string `gorm:"type:VARCHAR(100);index;" json:"District" yaml:"District,omitempty"`
|
|
|
|
PlaceCity string `gorm:"type:VARCHAR(100);index;" json:"City" yaml:"City,omitempty"`
|
|
|
|
PlaceState string `gorm:"type:VARCHAR(100);index;" json:"State" yaml:"State,omitempty"`
|
2020-09-13 17:51:43 +02:00
|
|
|
PlaceCountry string `gorm:"type:VARBINARY(2);" json:"Country" yaml:"Country,omitempty"`
|
2021-11-18 00:46:34 +01:00
|
|
|
PlaceKeywords string `gorm:"type:VARCHAR(300);" json:"Keywords" yaml:"Keywords,omitempty"`
|
2020-07-12 08:27:05 +02:00
|
|
|
PlaceFavorite bool `json:"Favorite" yaml:"Favorite,omitempty"`
|
|
|
|
PhotoCount int `gorm:"default:1" json:"PhotoCount" yaml:"-"`
|
|
|
|
CreatedAt time.Time `json:"CreatedAt" yaml:"-"`
|
|
|
|
UpdatedAt time.Time `json:"UpdatedAt" yaml:"-"`
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2021-11-12 05:09:17 +01:00
|
|
|
// TableName returns the entity database table name.
|
|
|
|
func (Place) TableName() string {
|
|
|
|
return "places"
|
|
|
|
}
|
|
|
|
|
2020-05-25 19:10:44 +02:00
|
|
|
// UnknownPlace is PhotoPrism's default place.
|
2020-04-25 14:22:47 +02:00
|
|
|
var UnknownPlace = Place{
|
2021-08-19 21:12:38 +02:00
|
|
|
ID: UnknownID,
|
2020-07-12 08:27:05 +02:00
|
|
|
PlaceLabel: "Unknown",
|
2021-11-12 05:09:17 +01:00
|
|
|
PlaceDistrict: "Unknown",
|
2021-11-18 00:46:34 +01:00
|
|
|
PlaceCity: "Unknown",
|
2020-07-12 08:27:05 +02:00
|
|
|
PlaceState: "Unknown",
|
2021-08-19 21:12:38 +02:00
|
|
|
PlaceCountry: UnknownID,
|
2020-07-12 08:27:05 +02:00
|
|
|
PlaceKeywords: "",
|
|
|
|
PlaceFavorite: false,
|
|
|
|
PhotoCount: -1,
|
2020-04-25 14:22:47 +02:00
|
|
|
}
|
2019-12-28 20:24:20 +01:00
|
|
|
|
2020-05-25 19:10:44 +02:00
|
|
|
// CreateUnknownPlace creates the default place if not exists.
|
2020-04-30 20:07:03 +02:00
|
|
|
func CreateUnknownPlace() {
|
2021-11-25 12:48:07 +01:00
|
|
|
UnknownPlace = *FirstOrCreatePlace(&UnknownPlace)
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-05-29 12:56:24 +02:00
|
|
|
// FindPlace finds a matching place or returns nil.
|
2021-11-20 16:36:34 +01:00
|
|
|
func FindPlace(id string) *Place {
|
2020-07-16 14:00:22 +02:00
|
|
|
place := Place{}
|
2019-12-28 20:24:20 +01:00
|
|
|
|
2021-11-20 16:36:34 +01:00
|
|
|
if err := Db().Where("id = ?", id).First(&place).Error; err != nil {
|
2022-04-15 09:42:07 +02:00
|
|
|
log.Debugf("place: %s not found", clean.Log(id))
|
2020-04-25 16:17:59 +02:00
|
|
|
return nil
|
2020-07-16 14:00:22 +02:00
|
|
|
} else {
|
|
|
|
return &place
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:21 +02:00
|
|
|
// Find fetches entity values from the database the primary key.
|
2020-04-30 20:07:03 +02:00
|
|
|
func (m *Place) Find() error {
|
2020-05-29 12:56:24 +02:00
|
|
|
if err := Db().First(m, "id = ?", m.ID).Error; err != nil {
|
2019-12-28 20:24:20 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-05-25 19:10:44 +02:00
|
|
|
// Create inserts a new row to the database.
|
|
|
|
func (m *Place) Create() error {
|
2020-12-13 15:43:01 +01:00
|
|
|
placeMutex.Lock()
|
|
|
|
defer placeMutex.Unlock()
|
|
|
|
|
2020-05-26 11:00:39 +02:00
|
|
|
return Db().Create(m).Error
|
2020-05-25 19:10:44 +02:00
|
|
|
}
|
|
|
|
|
2021-11-18 00:46:34 +01:00
|
|
|
// Save updates the existing or inserts a new row.
|
|
|
|
func (m *Place) Save() error {
|
|
|
|
placeMutex.Lock()
|
|
|
|
defer placeMutex.Unlock()
|
|
|
|
|
|
|
|
return Db().Save(m).Error
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete removes the entity from the index.
|
|
|
|
func (m *Place) Delete() (err error) {
|
|
|
|
return UnscopedDb().Delete(m).Error
|
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:21 +02:00
|
|
|
// FirstOrCreatePlace fetches an existing row, inserts a new row or nil in case of errors.
|
2020-05-25 19:10:44 +02:00
|
|
|
func FirstOrCreatePlace(m *Place) *Place {
|
2020-05-29 12:56:24 +02:00
|
|
|
if m.ID == "" {
|
2021-11-18 00:46:34 +01:00
|
|
|
log.Errorf("place: id must not be empty (find or create)")
|
2020-05-25 19:10:44 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-07-12 08:27:05 +02:00
|
|
|
if m.PlaceLabel == "" {
|
2021-11-18 00:46:34 +01:00
|
|
|
log.Errorf("place: label must not be empty (find or create place %s)", m.ID)
|
2020-05-25 19:10:44 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
result := Place{}
|
|
|
|
|
2021-11-20 16:36:34 +01:00
|
|
|
if findErr := Db().Where("id = ?", m.ID).First(&result).Error; findErr == nil {
|
2020-05-25 19:10:44 +02:00
|
|
|
return &result
|
2020-07-07 16:40:21 +02:00
|
|
|
} else if createErr := m.Create(); createErr == nil {
|
|
|
|
return m
|
2021-11-20 16:36:34 +01:00
|
|
|
} else if err := Db().Where("id = ?", m.ID).First(&result).Error; err == nil {
|
2020-07-07 16:40:21 +02:00
|
|
|
return &result
|
|
|
|
} else {
|
2021-11-18 00:46:34 +01:00
|
|
|
log.Errorf("place: %s (create %s)", createErr, m.ID)
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-07-07 16:40:21 +02:00
|
|
|
return nil
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-04-25 16:17:59 +02:00
|
|
|
// Unknown returns true if this is an unknown place
|
|
|
|
func (m Place) Unknown() bool {
|
2020-05-29 12:56:24 +02:00
|
|
|
return m.ID == "" || m.ID == UnknownPlace.ID
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// Label returns place label
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) Label() string {
|
2020-07-12 08:27:05 +02:00
|
|
|
return m.PlaceLabel
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2021-11-18 00:46:34 +01:00
|
|
|
// District returns the place district name if any.
|
|
|
|
func (m Place) District() string {
|
|
|
|
return m.PlaceDistrict
|
|
|
|
}
|
|
|
|
|
|
|
|
// City returns place city if any.
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) City() string {
|
2020-07-12 08:27:05 +02:00
|
|
|
return m.PlaceCity
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-05-26 21:51:34 +02:00
|
|
|
// LongCity checks if the city name is more than 16 char.
|
|
|
|
func (m Place) LongCity() bool {
|
2020-07-12 08:27:05 +02:00
|
|
|
return len(m.PlaceCity) > 16
|
2020-05-26 21:51:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NoCity checks if the location has no city
|
|
|
|
func (m Place) NoCity() bool {
|
2020-07-12 08:27:05 +02:00
|
|
|
return m.PlaceCity == ""
|
2020-05-26 21:51:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// CityContains checks if the location city contains the text string
|
|
|
|
func (m Place) CityContains(text string) bool {
|
2020-07-12 08:27:05 +02:00
|
|
|
return strings.Contains(text, m.PlaceCity)
|
2020-05-26 21:51:34 +02:00
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// State returns place State
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) State() string {
|
2020-07-12 08:27:05 +02:00
|
|
|
return m.PlaceState
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// CountryCode returns place CountryCode
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) CountryCode() string {
|
2020-07-12 08:27:05 +02:00
|
|
|
return m.PlaceCountry
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|
|
|
|
|
2020-02-21 01:14:45 +01:00
|
|
|
// CountryName returns place CountryName
|
2020-04-16 20:57:00 +02:00
|
|
|
func (m Place) CountryName() string {
|
2020-07-12 08:27:05 +02:00
|
|
|
return maps.CountryNames[m.PlaceCountry]
|
2019-12-28 20:24:20 +01:00
|
|
|
}
|