2013-12-24 03:09:50 +00:00
|
|
|
/*
|
|
|
|
Copyright 2013 The Camlistore Authors
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Package geocode handles mapping user-entered locations into lat/long polygons.
|
|
|
|
package geocode
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"io"
|
2013-12-24 21:43:25 +00:00
|
|
|
"log"
|
2013-12-24 03:09:50 +00:00
|
|
|
"net/url"
|
2013-12-24 20:43:19 +00:00
|
|
|
"sync"
|
2013-12-24 03:09:50 +00:00
|
|
|
|
|
|
|
"camlistore.org/pkg/context"
|
2014-01-29 06:00:52 +00:00
|
|
|
"camlistore.org/pkg/httputil"
|
2013-12-24 20:43:19 +00:00
|
|
|
"camlistore.org/pkg/singleflight"
|
2013-12-24 03:09:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type LatLong struct {
|
|
|
|
Lat float64 `json:"lat"`
|
|
|
|
Long float64 `json:"lng"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type Rect struct {
|
|
|
|
NorthEast LatLong `json:"northeast"`
|
|
|
|
SouthWest LatLong `json:"southwest"`
|
|
|
|
}
|
|
|
|
|
2013-12-24 20:43:19 +00:00
|
|
|
var (
|
|
|
|
mu sync.RWMutex
|
|
|
|
cache = map[string][]Rect{}
|
|
|
|
|
|
|
|
sf singleflight.Group
|
|
|
|
)
|
|
|
|
|
2013-12-24 03:09:50 +00:00
|
|
|
// Lookup returns rectangles for the given address. Currently the only
|
|
|
|
// implementation is the Google geocoding service.
|
|
|
|
func Lookup(ctx *context.Context, address string) ([]Rect, error) {
|
2013-12-24 20:43:19 +00:00
|
|
|
mu.RLock()
|
|
|
|
rects, ok := cache[address]
|
|
|
|
mu.RUnlock()
|
|
|
|
if ok {
|
|
|
|
return rects, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
rectsi, err := sf.Do(address, func() (interface{}, error) {
|
|
|
|
// TODO: static data files from OpenStreetMap, Wikipedia, etc?
|
|
|
|
urlStr := "https://maps.googleapis.com/maps/api/geocode/json?address=" + url.QueryEscape(address) + "&sensor=false"
|
2014-04-13 21:00:24 +00:00
|
|
|
res, err := ctx.HTTPClient().Get(urlStr)
|
2013-12-24 20:43:19 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2014-01-29 06:00:52 +00:00
|
|
|
defer httputil.CloseBody(res.Body)
|
2013-12-24 20:43:19 +00:00
|
|
|
rects, err := decodeGoogleResponse(res.Body)
|
2013-12-24 21:43:25 +00:00
|
|
|
log.Printf("Google geocode lookup (%q) = %#v, %v", address, rects, err)
|
2013-12-24 20:43:19 +00:00
|
|
|
if err == nil {
|
|
|
|
mu.Lock()
|
|
|
|
cache[address] = rects
|
|
|
|
mu.Unlock()
|
|
|
|
}
|
|
|
|
return rects, err
|
|
|
|
})
|
2013-12-24 03:09:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2013-12-24 20:43:19 +00:00
|
|
|
return rectsi.([]Rect), nil
|
2013-12-24 03:09:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type googleResTop struct {
|
|
|
|
Results []*googleResult `json:"results"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type googleResult struct {
|
|
|
|
Geometry *googleGeometry `json:"geometry"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type googleGeometry struct {
|
2014-01-25 17:26:18 +00:00
|
|
|
Bounds *Rect `json:"bounds"`
|
|
|
|
Viewport *Rect `json:"viewport"`
|
2013-12-24 03:09:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func decodeGoogleResponse(r io.Reader) (rects []Rect, err error) {
|
|
|
|
var resTop googleResTop
|
|
|
|
if err := json.NewDecoder(r).Decode(&resTop); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, res := range resTop.Results {
|
|
|
|
if res.Geometry != nil && res.Geometry.Bounds != nil {
|
2014-01-25 17:26:18 +00:00
|
|
|
r := res.Geometry.Bounds
|
|
|
|
if r.NorthEast.Lat == 90 && r.NorthEast.Long == 180 &&
|
|
|
|
r.SouthWest.Lat == -90 && r.SouthWest.Long == -180 {
|
|
|
|
// Google sometimes returns a "whole world" rect for large addresses (like "USA")
|
|
|
|
// so instead use the viewport in that case.
|
|
|
|
if res.Geometry.Viewport != nil {
|
|
|
|
rects = append(rects, *res.Geometry.Viewport)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
rects = append(rects, *r)
|
|
|
|
}
|
2013-12-24 03:09:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|