Annotate refreshment stops, improved logging and error handling.
This commit is contained in:
parent
7d3b208e5c
commit
52d41490f3
5 changed files with 195 additions and 25 deletions
|
@ -9,6 +9,9 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
|
||||||
|
"github.com/dhconnelly/rtreego"
|
||||||
|
|
||||||
|
"github.com/ray1729/gpx-utils/pkg/cafes"
|
||||||
"github.com/ray1729/gpx-utils/pkg/placenames"
|
"github.com/ray1729/gpx-utils/pkg/placenames"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -22,21 +25,26 @@ func main() {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
// TODO add --stops flag to select stops database
|
||||||
|
stopsIndex, err := cafes.FetchCtcCamIndex()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
gs, err := placenames.NewGPXSummarizer()
|
gs, err := placenames.NewGPXSummarizer()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
if info.IsDir() {
|
if info.IsDir() {
|
||||||
err = summarizeDirectory(gs, inFile)
|
err = summarizeDirectory(gs, stopsIndex, inFile)
|
||||||
} else {
|
} else {
|
||||||
err = summarizeSingleFile(gs, inFile)
|
err = summarizeSingleFile(gs, stopsIndex, inFile)
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func summarizeDirectory(gs *placenames.GPXSummarizer, dirName string) error {
|
func summarizeDirectory(gs *placenames.GPXSummarizer, stops *rtreego.Rtree, dirName string) error {
|
||||||
files, err := ioutil.ReadDir(dirName)
|
files, err := ioutil.ReadDir(dirName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -51,7 +59,7 @@ func summarizeDirectory(gs *placenames.GPXSummarizer, dirName string) error {
|
||||||
return fmt.Errorf("error opening %s for reading: %v", filename, err)
|
return fmt.Errorf("error opening %s for reading: %v", filename, err)
|
||||||
}
|
}
|
||||||
log.Printf("Analyzing %s", filename)
|
log.Printf("Analyzing %s", filename)
|
||||||
summary, err := gs.SummarizeTrack(r)
|
summary, err := gs.SummarizeTrack(r, stops)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("error creating summary of GPX track %s: %v", filename, err)
|
return fmt.Errorf("error creating summary of GPX track %s: %v", filename, err)
|
||||||
}
|
}
|
||||||
|
@ -72,12 +80,12 @@ func summarizeDirectory(gs *placenames.GPXSummarizer, dirName string) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func summarizeSingleFile(gs *placenames.GPXSummarizer, filename string) error {
|
func summarizeSingleFile(gs *placenames.GPXSummarizer, stops *rtreego.Rtree, filename string) error {
|
||||||
r, err := os.Open(filename)
|
r, err := os.Open(filename)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("error opening %s for reading: %v", filename, err)
|
return fmt.Errorf("error opening %s for reading: %v", filename, err)
|
||||||
}
|
}
|
||||||
summary, err := gs.SummarizeTrack(r)
|
summary, err := gs.SummarizeTrack(r, stops)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("error creating summary of GPX track %s: %v", filename, err)
|
return fmt.Errorf("error creating summary of GPX track %s: %v", filename, err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,11 +3,15 @@ package main
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/dhconnelly/rtreego"
|
||||||
|
|
||||||
|
"github.com/ray1729/gpx-utils/pkg/cafes"
|
||||||
"github.com/ray1729/gpx-utils/pkg/placenames"
|
"github.com/ray1729/gpx-utils/pkg/placenames"
|
||||||
"github.com/ray1729/gpx-utils/pkg/rwgps"
|
"github.com/ray1729/gpx-utils/pkg/rwgps"
|
||||||
)
|
)
|
||||||
|
@ -22,38 +26,73 @@ func main() {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
gpxSummarizer = gs
|
gpxSummarizer = gs
|
||||||
|
if err = loadStops(); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
http.HandleFunc("/rwgps", rwgpsHandler)
|
http.HandleFunc("/rwgps", rwgpsHandler)
|
||||||
|
log.Printf("Listening for requests on %s", listenAddr)
|
||||||
log.Fatal(http.ListenAndServe(listenAddr, nil))
|
log.Fatal(http.ListenAndServe(listenAddr, nil))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func loadStops() error {
|
||||||
|
var err error
|
||||||
|
log.Println("Fetching CTC Cambridge cafe stops")
|
||||||
|
stops["ctccam"], err = cafes.FetchCtcCamIndex()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
log.Printf("Loaded %d ctccam stops", stops["ctccam"].Size())
|
||||||
|
log.Println("Fetching cyclingmaps.net cafe stops")
|
||||||
|
stops["cyclingmapsnet"], err = cafes.FetchCyclingMapsIndex()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
log.Printf("Loaded %d cyclingmapsnet stops", stops["cyclingmapsnet"].Size())
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
var gpxSummarizer *placenames.GPXSummarizer
|
var gpxSummarizer *placenames.GPXSummarizer
|
||||||
|
var stops = make(map[string]*rtreego.Rtree)
|
||||||
|
|
||||||
func rwgpsHandler(w http.ResponseWriter, r *http.Request) {
|
func rwgpsHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
q := r.URL.Query()
|
q := r.URL.Query()
|
||||||
x := q.Get("routeId")
|
rawRouteId := q.Get("routeId")
|
||||||
log.Printf("Handilng request for routeId=%s", x)
|
stopsName := q.Get("stops")
|
||||||
if x == "" {
|
log.Printf("Handling request for routeId=%s stops=%s", rawRouteId, stopsName)
|
||||||
|
if rawRouteId == "" {
|
||||||
|
log.Printf("Missing routeId")
|
||||||
http.Error(w, "routeId is required", http.StatusBadRequest)
|
http.Error(w, "routeId is required", http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
routeId, err := strconv.Atoi(x)
|
routeId, err := strconv.Atoi(rawRouteId)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Invalid route id: %s", x)
|
log.Println("Error parsing route id '%s': %v", rawRouteId, err)
|
||||||
http.Error(w, "Invalid route id", http.StatusBadRequest)
|
http.Error(w, fmt.Sprintf("Invalid routeId: %s", rawRouteId), http.StatusBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
var stopsIndex *rtreego.Rtree
|
||||||
|
if stopsName != "" {
|
||||||
|
stopsIndex = stops[stopsName]
|
||||||
|
if stopsIndex == nil {
|
||||||
|
log.Printf("Invalid stops: %s", stopsName)
|
||||||
|
http.Error(w, fmt.Sprintf("Invalid stops: %s", stopsName), http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
track, err := rwgps.FetchTrack(routeId)
|
track, err := rwgps.FetchTrack(routeId)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println(err.Error())
|
log.Println(err.Error())
|
||||||
switch err.(type) {
|
switch err.(type) {
|
||||||
case *rwgps.ErrNotFound:
|
case *rwgps.ErrNotFound:
|
||||||
http.Error(w, err.Error(), http.StatusNotFound)
|
http.Error(w, err.Error(), http.StatusNotFound)
|
||||||
|
case *rwgps.ErrNotPublic:
|
||||||
|
http.Error(w, err.Error(), http.StatusForbidden)
|
||||||
default:
|
default:
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
summary, err := gpxSummarizer.SummarizeTrack(bytes.NewReader(track))
|
summary, err := gpxSummarizer.SummarizeTrack(bytes.NewReader(track), stopsIndex)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Error analyzing route %d: %v", routeId, err)
|
log.Printf("Error analyzing route %d: %v", routeId, err)
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
|
93
pkg/cafes/ctccam.go
Normal file
93
pkg/cafes/ctccam.go
Normal file
|
@ -0,0 +1,93 @@
|
||||||
|
package cafes
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/xml"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/dhconnelly/rtreego"
|
||||||
|
"github.com/fofanov/go-osgb"
|
||||||
|
)
|
||||||
|
|
||||||
|
const ctcCamWaypointsUrl = "https://ctccambridge.org.uk/ctccambridge-waypoints.gpx"
|
||||||
|
|
||||||
|
type Waypoint struct {
|
||||||
|
Lat float64 `xml:"lat,attr"`
|
||||||
|
Lon float64 `xml:"lon,attr"`
|
||||||
|
Name string `xml:"name"`
|
||||||
|
Url string `xml:"url"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Waypoints struct {
|
||||||
|
Waypoints []Waypoint `xml:"wpt"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type RefreshmentStop struct {
|
||||||
|
Name string
|
||||||
|
Url string
|
||||||
|
Easting float64
|
||||||
|
Northing float64
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *RefreshmentStop) Bounds() *rtreego.Rect {
|
||||||
|
p := rtreego.Point{s.Easting, s.Northing}
|
||||||
|
return p.ToRect(100)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *RefreshmentStop) Contains(p rtreego.Point) bool {
|
||||||
|
if len(p) != 2 {
|
||||||
|
panic("Expected a 2-dimensional point")
|
||||||
|
}
|
||||||
|
bounds := s.Bounds()
|
||||||
|
for i := 0; i < 2; i++ {
|
||||||
|
if p[i] < bounds.PointCoord(i) || p[i] > bounds.PointCoord(i)+bounds.LengthsCoord(i) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func BuildCtcCamIndex(r io.Reader) (*rtreego.Rtree, error) {
|
||||||
|
dec := xml.NewDecoder(r)
|
||||||
|
var wpt Waypoints
|
||||||
|
err := dec.Decode(&wpt)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
trans, err := osgb.NewOSTN15Transformer()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
stops := make([]rtreego.Spatial, len(wpt.Waypoints))
|
||||||
|
for i, w := range wpt.Waypoints {
|
||||||
|
gpsCoord := osgb.NewETRS89Coord(w.Lon, w.Lat, 0)
|
||||||
|
ngCoord, err := trans.ToNationalGrid(gpsCoord)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Error translating coordinates %v: %v", gpsCoord, err)
|
||||||
|
}
|
||||||
|
stops[i] = &RefreshmentStop{
|
||||||
|
Name: w.Name,
|
||||||
|
Url: w.Url,
|
||||||
|
Easting: ngCoord.Easting,
|
||||||
|
Northing: ngCoord.Northing,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return rtreego.NewTree(2, 25, 50, stops...), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func FetchCtcCamIndex() (*rtreego.Rtree, error) {
|
||||||
|
res, err := http.Get(ctcCamWaypointsUrl)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("error getting %s: %v", ctcCamWaypointsUrl, err)
|
||||||
|
}
|
||||||
|
defer res.Body.Close()
|
||||||
|
if res.StatusCode != http.StatusOK {
|
||||||
|
return nil, fmt.Errorf("unexpected status fetching %s: %s", ctcCamWaypointsUrl, res.Status)
|
||||||
|
}
|
||||||
|
index, err := BuildCtcCamIndex(res.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("error building CTC Cambridge stops index: %v", err)
|
||||||
|
}
|
||||||
|
return index, nil
|
||||||
|
}
|
|
@ -9,10 +9,12 @@ import (
|
||||||
"github.com/dhconnelly/rtreego"
|
"github.com/dhconnelly/rtreego"
|
||||||
"github.com/fofanov/go-osgb"
|
"github.com/fofanov/go-osgb"
|
||||||
"github.com/twpayne/go-gpx"
|
"github.com/twpayne/go-gpx"
|
||||||
|
|
||||||
|
"github.com/ray1729/gpx-utils/pkg/cafes"
|
||||||
)
|
)
|
||||||
|
|
||||||
type GPXSummarizer struct {
|
type GPXSummarizer struct {
|
||||||
rt *rtreego.Rtree
|
poi *rtreego.Rtree
|
||||||
trans osgb.CoordinateTransformer
|
trans osgb.CoordinateTransformer
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,7 +27,7 @@ func NewGPXSummarizer() (*GPXSummarizer, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return &GPXSummarizer{rt, trans}, nil
|
return &GPXSummarizer{poi: rt, trans: trans}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func distance(p1, p2 rtreego.Point) float64 {
|
func distance(p1, p2 rtreego.Point) float64 {
|
||||||
|
@ -46,6 +48,12 @@ type POI struct {
|
||||||
Distance float64
|
Distance float64
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type RefreshmentStop struct {
|
||||||
|
Name string
|
||||||
|
Url string
|
||||||
|
Distance float64
|
||||||
|
}
|
||||||
|
|
||||||
type TrackSummary struct {
|
type TrackSummary struct {
|
||||||
Name string
|
Name string
|
||||||
Time time.Time
|
Time time.Time
|
||||||
|
@ -55,9 +63,10 @@ type TrackSummary struct {
|
||||||
Distance float64
|
Distance float64
|
||||||
Ascent float64
|
Ascent float64
|
||||||
PointsOfInterest []POI
|
PointsOfInterest []POI
|
||||||
|
RefreshmentStops []RefreshmentStop `json:",omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gs *GPXSummarizer) SummarizeTrack(r io.Reader) (*TrackSummary, error) {
|
func (gs *GPXSummarizer) SummarizeTrack(r io.Reader, stops *rtreego.Rtree) (*TrackSummary, error) {
|
||||||
g, err := gpx.Read(r)
|
g, err := gpx.Read(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -76,6 +85,7 @@ func (gs *GPXSummarizer) SummarizeTrack(r io.Reader) (*TrackSummary, error) {
|
||||||
var prevPlacePoint rtreego.Point
|
var prevPlacePoint rtreego.Point
|
||||||
var prevPoint rtreego.Point
|
var prevPoint rtreego.Point
|
||||||
var prevHeight float64
|
var prevHeight float64
|
||||||
|
var prevStop *cafes.RefreshmentStop
|
||||||
|
|
||||||
init := true
|
init := true
|
||||||
for _, trk := range g.Trk {
|
for _, trk := range g.Trk {
|
||||||
|
@ -88,7 +98,7 @@ func (gs *GPXSummarizer) SummarizeTrack(r io.Reader) (*TrackSummary, error) {
|
||||||
}
|
}
|
||||||
thisPoint := rtreego.Point{ngCoord.Easting, ngCoord.Northing}
|
thisPoint := rtreego.Point{ngCoord.Easting, ngCoord.Northing}
|
||||||
thisHeight := ngCoord.Height
|
thisHeight := ngCoord.Height
|
||||||
nn, _ := gs.rt.NearestNeighbor(thisPoint).(*NamedBoundary)
|
nn, _ := gs.poi.NearestNeighbor(thisPoint).(*NamedBoundary)
|
||||||
if init {
|
if init {
|
||||||
s.Start = nn.Name
|
s.Start = nn.Name
|
||||||
prevPlace = nn.Name
|
prevPlace = nn.Name
|
||||||
|
@ -108,6 +118,17 @@ func (gs *GPXSummarizer) SummarizeTrack(r io.Reader) (*TrackSummary, error) {
|
||||||
prevPlace = nn.Name
|
prevPlace = nn.Name
|
||||||
prevPlacePoint = thisPoint
|
prevPlacePoint = thisPoint
|
||||||
}
|
}
|
||||||
|
if stops != nil {
|
||||||
|
stop, ok := stops.NearestNeighbor(thisPoint).(*cafes.RefreshmentStop)
|
||||||
|
if ok && stop.Contains(thisPoint) && (prevStop == nil || stop.Name != prevStop.Name) {
|
||||||
|
s.RefreshmentStops = append(s.RefreshmentStops, RefreshmentStop{
|
||||||
|
Name: stop.Name,
|
||||||
|
Url: stop.Url,
|
||||||
|
Distance: s.Distance,
|
||||||
|
})
|
||||||
|
prevStop = stop
|
||||||
|
}
|
||||||
|
}
|
||||||
prevPoint = thisPoint
|
prevPoint = thisPoint
|
||||||
prevHeight = thisHeight
|
prevHeight = thisHeight
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package rwgps
|
package rwgps
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
@ -15,6 +14,14 @@ func (e *ErrNotFound) Error() string {
|
||||||
return fmt.Sprintf("RideWithGPS track %d not found", e.RouteId)
|
return fmt.Sprintf("RideWithGPS track %d not found", e.RouteId)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ErrNotPublic struct {
|
||||||
|
RouteId int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *ErrNotPublic) Error() string {
|
||||||
|
return fmt.Sprintf("RideWithGPS track %d is not public", e.RouteId)
|
||||||
|
}
|
||||||
|
|
||||||
func FetchTrack(routeId int) ([]byte, error) {
|
func FetchTrack(routeId int) ([]byte, error) {
|
||||||
url := fmt.Sprintf("https://ridewithgps.com/routes/%d.gpx?sub_format=track", routeId)
|
url := fmt.Sprintf("https://ridewithgps.com/routes/%d.gpx?sub_format=track", routeId)
|
||||||
resp, err := http.Get(url)
|
resp, err := http.Get(url)
|
||||||
|
@ -22,16 +29,18 @@ func FetchTrack(routeId int) ([]byte, error) {
|
||||||
return nil, fmt.Errorf("error getting %s: %v", url, err)
|
return nil, fmt.Errorf("error getting %s: %v", url, err)
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
if resp.StatusCode == http.StatusNotFound {
|
||||||
|
return nil, &ErrNotFound{routeId}
|
||||||
|
}
|
||||||
|
if resp.StatusCode == http.StatusForbidden {
|
||||||
|
return nil, &ErrNotPublic{routeId}
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("error retrieving route %d: %s", routeId, resp.Status)
|
||||||
|
}
|
||||||
data, err := ioutil.ReadAll(resp.Body)
|
data, err := ioutil.ReadAll(resp.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("error reading response from %s: %v", url, err)
|
return nil, fmt.Errorf("error reading response from %s: %v", url, err)
|
||||||
}
|
}
|
||||||
if IsNotFound(data) {
|
|
||||||
return nil, &ErrNotFound{routeId}
|
|
||||||
}
|
|
||||||
return data, nil
|
return data, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func IsNotFound(data []byte) bool {
|
|
||||||
return bytes.HasPrefix(data, []byte("<!DOCTYPE html>")) && bytes.Contains(data, []byte("Error (404 not found)"))
|
|
||||||
}
|
|
||||||
|
|
Loading…
Reference in a new issue