package api import ( "net/http" "github.com/photoprism/photoprism/internal/acl" "github.com/photoprism/photoprism/internal/entity" "github.com/photoprism/photoprism/internal/form" "github.com/photoprism/photoprism/internal/search" "github.com/photoprism/photoprism/pkg/txt" "github.com/gin-gonic/gin" "github.com/gin-gonic/gin/binding" geojson "github.com/paulmach/go.geojson" ) // SearchPhotosGeo performs a geo search with reduced metadata details for improved performance. // // GET /api/v1/geo func SearchPhotosGeo(router *gin.RouterGroup) { router.GET("/geo", func(c *gin.Context) { s := Auth(SessionID(c), acl.ResourcePhotos, acl.ActionSearch) if s.Invalid() { AbortUnauthorized(c) return } var f form.GeoSearch err := c.MustBindWith(&f, binding.Form) if err != nil { AbortBadRequest(c) return } photos, err := search.PhotosGeo(f) if err != nil { log.Warnf("search: %s", err) AbortBadRequest(c) return } fc := geojson.NewFeatureCollection() bbox := make([]float64, 4) bboxMin := func(pos int, val float64) { if bbox[pos] == 0.0 || bbox[pos] > val { bbox[pos] = val } } bboxMax := func(pos int, val float64) { if bbox[pos] == 0.0 || bbox[pos] < val { bbox[pos] = val } } for _, p := range photos { bboxMin(0, p.Lng()) bboxMin(1, p.Lat()) bboxMax(2, p.Lng()) bboxMax(3, p.Lat()) props := gin.H{ "UID": p.PhotoUID, "Hash": p.FileHash, "Width": p.FileWidth, "Height": p.FileHeight, "TakenAt": p.TakenAt, "Title": p.PhotoTitle, } if p.PhotoDescription != "" { props["Description"] = p.PhotoDescription } if p.PhotoType != entity.TypeImage && p.PhotoType != entity.TypeDefault { props["Type"] = p.PhotoType } if p.PhotoFavorite { props["Favorite"] = true } feat := geojson.NewPointFeature([]float64{p.Lng(), p.Lat()}) feat.ID = p.ID feat.Properties = props fc.AddFeature(feat) } fc.BoundingBox = bbox resp, err := fc.MarshalJSON() if err != nil { c.AbortWithStatusJSON(400, gin.H{"error": txt.UcFirst(err.Error())}) return } AddTokenHeaders(c) c.Data(http.StatusOK, "application/json", resp) }) }