photoprism/internal/hub/places/location.go

218 lines
5.1 KiB
Go

package places
import (
"crypto/sha1"
"encoding/json"
"fmt"
"net/http"
"strings"
"time"
"github.com/photoprism/photoprism/pkg/s2"
"github.com/photoprism/photoprism/pkg/sanitize"
"github.com/photoprism/photoprism/pkg/txt"
)
// Location represents a specific geolocation identified by its S2 ID.
type Location struct {
ID string `json:"id"`
LocLat float64 `json:"lat"`
LocLng float64 `json:"lng"`
LocName string `json:"name"`
LocStreet string `json:"street"`
LocPostcode string `json:"postcode"`
LocCategory string `json:"category"`
Place Place `json:"place"`
Cached bool `json:"-"`
}
// ApiName is the backend API name.
const ApiName = "places"
var Key = "f60f5b25d59c397989e3cd374f81cdd7710a4fca"
var Secret = "photoprism"
var UserAgent = "PhotoPrism/dev"
var ReverseLookupURL = "https://places.photoprism.app/v1/location/%s"
var Retries = 3
var RetryDelay = 33 * time.Millisecond
var client = &http.Client{Timeout: 60 * time.Second}
// FindLocation retrieves location details from the backend API.
func FindLocation(id string) (result Location, err error) {
// Normalize S2 Cell ID.
id = s2.NormalizeToken(id)
// Valid?
if len(id) == 0 {
return result, fmt.Errorf("empty cell id")
} else if n := len(id); n < 4 || n > 16 {
return result, fmt.Errorf("invalid cell id %s", sanitize.Log(id))
}
// Remember start time.
start := time.Now()
// Convert S2 Cell ID to latitude and longitude.
lat, lng := s2.LatLng(id)
// Return if latitude and longitude are null.
if lat == 0.0 || lng == 0.0 {
return result, fmt.Errorf("skipping lat %f, lng %f", lat, lng)
}
// Location details cached?
if hit, ok := cache.Get(id); ok {
log.Tracef("places: cache hit for lat %f, lng %f", lat, lng)
cached := hit.(Location)
cached.Cached = true
return cached, nil
}
// Compose request URL.
url := fmt.Sprintf(ReverseLookupURL, id)
// Log request URL.
log.Tracef("places: sending request to %s", url)
// Create GET request instance.
req, err := http.NewRequest(http.MethodGet, url, nil)
// Ok?
if err != nil {
log.Errorf("places: %s", err.Error())
return result, err
}
// Add User-Agent header?
if UserAgent != "" {
req.Header.Set("User-Agent", UserAgent)
}
// Add API key?
if Key != "" {
req.Header.Set("X-Key", Key)
req.Header.Set("X-Signature", fmt.Sprintf("%x", sha1.Sum([]byte(Key+url+Secret))))
}
var r *http.Response
// Perform request.
for i := 0; i < Retries; i++ {
r, err = client.Do(req)
// Successful?
if err == nil {
break
}
// Wait before trying again?
if RetryDelay.Nanoseconds() > 0 {
time.Sleep(RetryDelay)
}
}
// Failed?
if err != nil {
log.Errorf("places: %s (http request failed)", err.Error())
return result, err
} else if r.StatusCode >= 400 {
err = fmt.Errorf("request failed with code %d", r.StatusCode)
return result, err
}
// Decode JSON response body.
err = json.NewDecoder(r.Body).Decode(&result)
if err != nil {
log.Errorf("places: %s (decode json failed)", err.Error())
return result, err
}
if result.ID == "" {
return result, fmt.Errorf("no result for %s", id)
}
cache.SetDefault(id, result)
log.Tracef("places: cached cell %s [%s]", sanitize.Log(id), time.Since(start))
result.Cached = false
return result, nil
}
// CellID returns the S2 cell identifier string.
func (l Location) CellID() string {
return l.ID
}
// PlaceID returns the place identifier string.
func (l Location) PlaceID() string {
return l.Place.PlaceID
}
// Name returns the location name if any.
func (l Location) Name() (result string) {
return strings.SplitN(l.LocName, "/", 2)[0]
}
// Street returns the location street if any.
func (l Location) Street() (result string) {
return strings.SplitN(l.LocStreet, "/", 2)[0]
}
// Postcode returns the location postcode if any.
func (l Location) Postcode() (result string) {
return strings.SplitN(l.LocPostcode, "/", 2)[0]
}
// Category returns the location category if any.
func (l Location) Category() (result string) {
return l.LocCategory
}
// Label returns the location label.
func (l Location) Label() (result string) {
return l.Place.LocLabel
}
// City returns the location address city name.
func (l Location) City() (result string) {
return l.Place.LocCity
}
// District returns the location address district name.
func (l Location) District() (result string) {
return l.Place.LocDistrict
}
// CountryCode returns the location address country code.
func (l Location) CountryCode() (result string) {
return l.Place.LocCountry
}
// State returns the location address state name.
func (l Location) State() (result string) {
return sanitize.State(l.Place.LocState, l.CountryCode())
}
// Latitude returns the location position latitude.
func (l Location) Latitude() (result float64) {
return l.LocLat
}
// Longitude returns the location position longitude.
func (l Location) Longitude() (result float64) {
return l.LocLng
}
// Keywords returns location keywords if any.
func (l Location) Keywords() (result []string) {
return txt.UniqueWords(txt.Words(l.Place.LocKeywords))
}
// Source returns the backend API name.
func (l Location) Source() string {
return "places"
}