60efc86649
Signed-off-by: Michael Mayer <michael@photoprism.app>
162 lines
3.5 KiB
Go
162 lines
3.5 KiB
Go
package meta
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"runtime/debug"
|
|
"time"
|
|
|
|
"github.com/photoprism/photoprism/pkg/txt"
|
|
"gopkg.in/photoprism/go-tz.v2/tz"
|
|
)
|
|
|
|
type GPhoto struct {
|
|
Title string `json:"title"`
|
|
Description string `json:"description"`
|
|
Views int `json:"imageViews,string"`
|
|
Geo GGeo `json:"geoData"`
|
|
TakenAt GTime `json:"photoTakenTime"`
|
|
CreatedAt GTime `json:"creationTime"`
|
|
UpdatedAt GTime `json:"modificationTime"`
|
|
}
|
|
|
|
func (m GPhoto) SanitizedTitle() string {
|
|
return SanitizeTitle(m.Title)
|
|
}
|
|
|
|
func (m GPhoto) SanitizedDescription() string {
|
|
return SanitizeDescription(m.Description)
|
|
}
|
|
|
|
type GMeta struct {
|
|
Album GAlbum `json:"albumData"`
|
|
}
|
|
|
|
type GAlbum struct {
|
|
Title string `json:"title"`
|
|
Description string `json:"description"`
|
|
Access string `json:"access"`
|
|
Location string `json:"location"`
|
|
Date GTime `json:"date"`
|
|
Geo GGeo `json:"geoData"`
|
|
}
|
|
|
|
func (m GAlbum) Exists() bool {
|
|
return m.Title != ""
|
|
}
|
|
|
|
type GGeo struct {
|
|
Lat float64 `json:"latitude"`
|
|
Lng float64 `json:"longitude"`
|
|
Altitude float64 `json:"altitude"`
|
|
}
|
|
|
|
func (m GGeo) Exists() bool {
|
|
return m.Lat != 0.0 && m.Lng != 0.0
|
|
}
|
|
|
|
type GTime struct {
|
|
Unix int64 `json:"timestamp,string"`
|
|
Formatted string `json:"formatted"`
|
|
}
|
|
|
|
func (m GTime) Exists() bool {
|
|
return m.Unix > 0
|
|
}
|
|
|
|
func (m GTime) Time() time.Time {
|
|
return time.Unix(m.Unix, 0).UTC()
|
|
}
|
|
|
|
// GMeta parses JSON sidecar data as created by Google Photos.
|
|
func (data *Data) GMeta(jsonData []byte) (err error) {
|
|
defer func() {
|
|
if e := recover(); e != nil {
|
|
err = fmt.Errorf("metadata: %s (gmeta panic)\nstack: %s", e, debug.Stack())
|
|
}
|
|
}()
|
|
|
|
p := GMeta{}
|
|
|
|
if err := json.Unmarshal(jsonData, &p); err != nil {
|
|
return err
|
|
}
|
|
|
|
if p.Album.Exists() {
|
|
data.Albums = append(data.Albums, p.Album.Title)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// GPhoto parses JSON photo sidecar data as created by Google Photos.
|
|
func (data *Data) GPhoto(jsonData []byte) (err error) {
|
|
defer func() {
|
|
if e := recover(); e != nil {
|
|
err = fmt.Errorf("metadata: %s (gphoto panic)\nstack: %s", e, debug.Stack())
|
|
}
|
|
}()
|
|
|
|
p := GPhoto{}
|
|
|
|
if err := json.Unmarshal(jsonData, &p); err != nil {
|
|
return err
|
|
}
|
|
|
|
if s := p.SanitizedTitle(); s != "" && data.Title == "" {
|
|
data.Title = s
|
|
}
|
|
|
|
if s := p.SanitizedDescription(); s != "" && data.Description == "" {
|
|
data.Description = s
|
|
}
|
|
|
|
if p.Views > 0 && data.Views == 0 {
|
|
data.Views = p.Views
|
|
}
|
|
|
|
if p.TakenAt.Exists() {
|
|
if data.TakenAt.IsZero() {
|
|
data.TakenAt = p.TakenAt.Time()
|
|
}
|
|
|
|
if data.TakenAtLocal.IsZero() {
|
|
data.TakenAtLocal = p.TakenAt.Time()
|
|
}
|
|
}
|
|
|
|
if p.Geo.Exists() {
|
|
if data.Lat == 0 && data.Lng == 0 {
|
|
data.Lat = float32(p.Geo.Lat)
|
|
data.Lng = float32(p.Geo.Lng)
|
|
}
|
|
|
|
if data.Altitude == 0 {
|
|
data.Altitude = p.Geo.Altitude
|
|
}
|
|
}
|
|
|
|
// Set time zone and calculate UTC time.
|
|
if data.Lat != 0 && data.Lng != 0 {
|
|
zones, err := tz.GetZone(tz.Point{
|
|
Lat: float64(data.Lat),
|
|
Lon: float64(data.Lng),
|
|
})
|
|
|
|
if err == nil && len(zones) > 0 {
|
|
data.TimeZone = zones[0]
|
|
}
|
|
|
|
if !data.TakenAtLocal.IsZero() {
|
|
if loc := txt.TimeZone(data.TimeZone); loc == nil {
|
|
log.Warnf("metadata: invalid time zone %s (gphotos)", data.TimeZone)
|
|
} else if tl, err := time.ParseInLocation("2006:01:02 15:04:05", data.TakenAtLocal.Format("2006:01:02 15:04:05"), loc); err == nil {
|
|
data.TakenAt = tl.UTC().Truncate(time.Second)
|
|
} else {
|
|
log.Errorf("metadata: %s (gphotos)", err.Error()) // this should never happen
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|