mirror of
https://github.com/kevin-DL/services.git
synced 2026-01-11 19:04:35 +00:00
273 lines
6.3 KiB
Go
273 lines
6.3 KiB
Go
package handler
|
|
|
|
import (
|
|
"context"
|
|
"net/url"
|
|
"time"
|
|
|
|
"github.com/gojuno/go.osrm"
|
|
"github.com/micro/micro/v3/service/errors"
|
|
pb "github.com/micro/services/routing/proto"
|
|
"github.com/paulmach/go.geo"
|
|
)
|
|
|
|
type OSRM struct {
|
|
// api address
|
|
Address string
|
|
// osrm client
|
|
Client *osrm.OSRM
|
|
}
|
|
|
|
func (o *OSRM) Directions(ctx context.Context, req *pb.DirectionsRequest, rsp *pb.DirectionsResponse) error {
|
|
// validate the request
|
|
if req.Origin == nil {
|
|
return ErrMissingOrigin
|
|
}
|
|
if req.Destination == nil {
|
|
return ErrMissingDestination
|
|
}
|
|
if err := validatePoint(req.Origin); err != nil {
|
|
return err
|
|
}
|
|
if err := validatePoint(req.Destination); err != nil {
|
|
return err
|
|
}
|
|
|
|
orig := req.Origin
|
|
dest := req.Destination
|
|
|
|
if o.Client == nil {
|
|
u, _ := url.Parse(o.Address)
|
|
if u.Scheme == "" {
|
|
u.Scheme = "http"
|
|
}
|
|
o.Client = osrm.NewFromURL(u.String())
|
|
}
|
|
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), time.Second)
|
|
defer cancelFn()
|
|
|
|
resp, err := o.Client.Route(ctx, osrm.RouteRequest{
|
|
Profile: "car",
|
|
Coordinates: osrm.NewGeometryFromPointSet(geo.PointSet{
|
|
{orig.Longitude, orig.Latitude},
|
|
{dest.Longitude, dest.Latitude},
|
|
}),
|
|
Steps: osrm.StepsTrue,
|
|
Annotations: osrm.AnnotationsFalse,
|
|
Overview: osrm.OverviewFalse,
|
|
})
|
|
if err != nil {
|
|
return errors.InternalServerError("routing.eta", "failed to get route: %v", err.Error())
|
|
}
|
|
|
|
if len(resp.Routes) == 0 {
|
|
return nil
|
|
}
|
|
|
|
// set the estimated duration and distance
|
|
rsp.Duration = float64(resp.Routes[0].Duration)
|
|
rsp.Distance = float64(resp.Routes[0].Distance)
|
|
|
|
for _, leg := range resp.Routes[0].Legs {
|
|
for _, step := range leg.Steps {
|
|
// set the waypoints for the route
|
|
for _, intersect := range step.Intersections {
|
|
rsp.Waypoints = append(rsp.Waypoints, &pb.Waypoint{
|
|
Name: step.Name,
|
|
Location: &pb.Point{
|
|
Latitude: intersect.Location.Lat(),
|
|
Longitude: intersect.Location.Lng(),
|
|
},
|
|
})
|
|
}
|
|
|
|
instruction := step.Maneuver.Modifier
|
|
|
|
switch step.Maneuver.Type {
|
|
case "new name":
|
|
instruction = "go " + step.Maneuver.Modifier
|
|
default:
|
|
instruction = step.Maneuver.Type + " " + step.Maneuver.Modifier
|
|
}
|
|
|
|
var intersections []*pb.Intersection
|
|
|
|
for _, is := range step.Intersections {
|
|
var bearings []float64
|
|
|
|
for _, bearing := range is.Bearings {
|
|
bearings = append(bearings, float64(bearing))
|
|
}
|
|
|
|
intersections = append(intersections, &pb.Intersection{
|
|
Bearings: bearings,
|
|
Location: &pb.Point{
|
|
Latitude: is.Location.Lat(),
|
|
Longitude: is.Location.Lng(),
|
|
},
|
|
})
|
|
}
|
|
|
|
action := step.Maneuver.Type
|
|
if action == "new name" {
|
|
action = "continue"
|
|
}
|
|
|
|
// set the directions for the route
|
|
rsp.Directions = append(rsp.Directions, &pb.Direction{
|
|
Name: step.Name,
|
|
Instruction: instruction,
|
|
Distance: float64(step.Distance),
|
|
Duration: float64(step.Duration),
|
|
Maneuver: &pb.Maneuver{
|
|
BearingBefore: float64(step.Maneuver.BearingBefore),
|
|
BearingAfter: float64(step.Maneuver.BearingAfter),
|
|
Location: &pb.Point{
|
|
Latitude: float64(step.Maneuver.Location.Lat()),
|
|
Longitude: float64(step.Maneuver.Location.Lng()),
|
|
},
|
|
Action: action,
|
|
Direction: step.Maneuver.Modifier,
|
|
},
|
|
Intersections: intersections,
|
|
})
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (o *OSRM) Eta(ctx context.Context, req *pb.EtaRequest, rsp *pb.EtaResponse) error {
|
|
// validate the request
|
|
if req.Origin == nil {
|
|
return ErrMissingOrigin
|
|
}
|
|
if req.Destination == nil {
|
|
return ErrMissingDestination
|
|
}
|
|
if err := validatePoint(req.Origin); err != nil {
|
|
return err
|
|
}
|
|
if err := validatePoint(req.Destination); err != nil {
|
|
return err
|
|
}
|
|
|
|
orig := req.Origin
|
|
dest := req.Destination
|
|
|
|
if o.Client == nil {
|
|
u, _ := url.Parse(o.Address)
|
|
if u.Scheme == "" {
|
|
u.Scheme = "http"
|
|
}
|
|
o.Client = osrm.NewFromURL(u.String())
|
|
}
|
|
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), time.Second)
|
|
defer cancelFn()
|
|
|
|
resp, err := o.Client.Route(ctx, osrm.RouteRequest{
|
|
Profile: "car",
|
|
Coordinates: osrm.NewGeometryFromPointSet(geo.PointSet{
|
|
{orig.Longitude, orig.Latitude},
|
|
{dest.Longitude, dest.Latitude},
|
|
}),
|
|
Steps: osrm.StepsFalse,
|
|
Annotations: osrm.AnnotationsFalse,
|
|
Overview: osrm.OverviewFalse,
|
|
})
|
|
if err != nil {
|
|
return errors.InternalServerError("routing.eta", "failed to get route: %v", err.Error())
|
|
}
|
|
|
|
if len(resp.Routes) == 0 {
|
|
return nil
|
|
}
|
|
|
|
// distance is meters
|
|
distance := resp.Routes[0].Distance
|
|
// duration is seconds
|
|
duration := resp.Routes[0].Duration
|
|
|
|
// nothing to calculate
|
|
if distance == 0.0 {
|
|
return nil
|
|
}
|
|
|
|
// set the duration
|
|
rsp.Duration = float64(duration)
|
|
|
|
// TODO: calculate transport/speed
|
|
|
|
return nil
|
|
}
|
|
|
|
func (o *OSRM) Route(ctx context.Context, req *pb.RouteRequest, rsp *pb.RouteResponse) error {
|
|
// validate the request
|
|
if req.Origin == nil {
|
|
return ErrMissingOrigin
|
|
}
|
|
if req.Destination == nil {
|
|
return ErrMissingDestination
|
|
}
|
|
if err := validatePoint(req.Origin); err != nil {
|
|
return err
|
|
}
|
|
if err := validatePoint(req.Destination); err != nil {
|
|
return err
|
|
}
|
|
|
|
if o.Client == nil {
|
|
u, _ := url.Parse(o.Address)
|
|
if u.Scheme == "" {
|
|
u.Scheme = "http"
|
|
}
|
|
o.Client = osrm.NewFromURL(u.String())
|
|
}
|
|
|
|
ctx, cancelFn := context.WithTimeout(context.Background(), time.Second)
|
|
defer cancelFn()
|
|
|
|
orig := req.Origin
|
|
dest := req.Destination
|
|
|
|
resp, err := o.Client.Route(ctx, osrm.RouteRequest{
|
|
Profile: "car",
|
|
Coordinates: osrm.NewGeometryFromPointSet(geo.PointSet{
|
|
{orig.Longitude, orig.Latitude},
|
|
{dest.Longitude, dest.Latitude},
|
|
}),
|
|
Steps: osrm.StepsTrue,
|
|
Annotations: osrm.AnnotationsFalse,
|
|
Overview: osrm.OverviewFalse,
|
|
})
|
|
if err != nil {
|
|
return errors.InternalServerError("routing.route", "failed to get route: %v", err.Error())
|
|
}
|
|
|
|
if len(resp.Routes) == 0 {
|
|
return nil
|
|
}
|
|
|
|
// set distance and duration
|
|
rsp.Duration = float64(resp.Routes[0].Duration)
|
|
rsp.Distance = float64(resp.Routes[0].Distance)
|
|
|
|
for _, leg := range resp.Routes[0].Legs {
|
|
for _, step := range leg.Steps {
|
|
for _, intersect := range step.Intersections {
|
|
rsp.Waypoints = append(rsp.Waypoints, &pb.Waypoint{
|
|
Name: step.Name,
|
|
Location: &pb.Point{
|
|
Latitude: intersect.Location.Lat(),
|
|
Longitude: intersect.Location.Lng(),
|
|
},
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|