diff --git a/services/travelmux/src/api/mod.rs b/services/travelmux/src/api/mod.rs index 1d6fdf9f8..d9d9c1a38 100644 --- a/services/travelmux/src/api/mod.rs +++ b/services/travelmux/src/api/mod.rs @@ -4,3 +4,4 @@ pub use app_state::AppState; pub mod health; pub mod v4; pub mod v5; +pub mod v6; diff --git a/services/travelmux/src/api/v5/error.rs b/services/travelmux/src/api/v5/error.rs index 8e33623e8..d300f1af4 100644 --- a/services/travelmux/src/api/v5/error.rs +++ b/services/travelmux/src/api/v5/error.rs @@ -2,7 +2,7 @@ use crate::otp::otp_api; use crate::valhalla::valhalla_api; use crate::{DistanceUnit, Error, TravelMode}; use actix_web::HttpResponseBuilder; -use serde::Serialize; +use serde::{Deserialize, Serialize}; use super::{Itinerary, Plan}; use crate::error::ErrorType; @@ -32,7 +32,7 @@ pub struct PlanError { pub message: String, } -#[derive(Debug, Serialize, Clone, PartialEq)] +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] #[serde(rename_all = "camelCase")] pub struct PlanResponseOk { pub(crate) plan: Plan, diff --git a/services/travelmux/src/api/v5/mod.rs b/services/travelmux/src/api/v5/mod.rs index 9496d312e..20c8b18b4 100644 --- a/services/travelmux/src/api/v5/mod.rs +++ b/services/travelmux/src/api/v5/mod.rs @@ -1,10 +1,4 @@ -pub mod directions; mod error; -mod haversine_segmenter; -mod osrm_api; pub mod plan; -mod travel_modes; - -pub use travel_modes::TravelModes; pub use plan::{Itinerary, Plan}; diff --git a/services/travelmux/src/api/v5/plan.rs b/services/travelmux/src/api/v5/plan.rs index 50ece2e60..80746b1f3 100644 --- a/services/travelmux/src/api/v5/plan.rs +++ b/services/travelmux/src/api/v5/plan.rs @@ -3,44 +3,26 @@ use geo::algorithm::BoundingRect; use geo::geometry::{LineString, Point, Rect}; use polyline::decode_polyline; use reqwest::header::{HeaderName, HeaderValue}; -use serde::{Deserialize, Serialize}; +use serde::{de, de::IntoDeserializer, de::Visitor, Deserialize, Deserializer, Serialize}; use std::collections::HashMap; +use std::fmt; use std::time::{Duration, SystemTime}; use super::error::{PlanResponseErr, PlanResponseOk}; -use super::TravelModes; - -use crate::api::v5::haversine_segmenter::HaversineSegmenter; use crate::api::AppState; use crate::error::ErrorType; use crate::otp::otp_api; use crate::otp::otp_api::{AbsoluteDirection, RelativeDirection}; use crate::util::format::format_meters; use crate::util::{ - deserialize_point_from_lat_lon, extend_bounds, serialize_line_string_as_polyline6, - serialize_rect_to_lng_lat, serialize_system_time_as_millis, system_time_from_millis, + deserialize_point_from_lat_lon, extend_bounds, serialize_rect_to_lng_lat, + serialize_system_time_as_millis, system_time_from_millis, }; use crate::valhalla::valhalla_api; use crate::valhalla::valhalla_api::{LonLat, ManeuverType}; use crate::{DistanceUnit, Error, TravelMode}; -const METERS_PER_MILE: f64 = 1609.34; - -fn convert_from_meters(meters: f64, output_units: DistanceUnit) -> f64 { - match output_units { - DistanceUnit::Kilometers => meters / 1000.0, - DistanceUnit::Miles => meters / METERS_PER_MILE, - } -} - -fn convert_to_meters(distance: f64, input_units: DistanceUnit) -> f64 { - match input_units { - DistanceUnit::Kilometers => distance * 1000.0, - DistanceUnit::Miles => distance * METERS_PER_MILE, - } -} - -#[derive(Debug, Deserialize, Clone, PartialEq)] +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] #[serde(rename_all = "camelCase")] pub struct PlanQuery { #[serde(deserialize_with = "deserialize_point_from_lat_lon")] @@ -58,51 +40,32 @@ pub struct PlanQuery { preferred_distance_units: Option, } -#[derive(Debug, Serialize, Clone, PartialEq)] +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] #[serde(rename_all = "camelCase")] pub struct Plan { pub(crate) itineraries: Vec, } -#[derive(Debug, Serialize, Clone, PartialEq)] +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] #[serde(rename_all = "camelCase")] pub struct Itinerary { mode: TravelMode, /// seconds - pub(crate) duration: f64, + duration: f64, /// unix millis, UTC #[serde(serialize_with = "serialize_system_time_as_millis")] start_time: SystemTime, /// unix millis, UTC #[serde(serialize_with = "serialize_system_time_as_millis")] end_time: SystemTime, - /// Units are in `distance_units` distance: f64, - /// FIXME: I think we're returning meters even though distance unit is "Kilometers" - /// Probably we should rename DistanceUnit::Kilometers to DistanceUnit::Meters - /// This is passed as a parameter though, so it'd be a breaking change. - pub(crate) distance_units: DistanceUnit, + distance_units: DistanceUnit, #[serde(serialize_with = "serialize_rect_to_lng_lat")] bounds: Rect, - pub(crate) legs: Vec, + legs: Vec, } impl Itinerary { - pub fn distance_meters(&self) -> f64 { - match self.distance_units { - DistanceUnit::Kilometers => self.distance * 1000.0, - DistanceUnit::Miles => self.distance * METERS_PER_MILE, - } - } - - pub fn combined_geometry(&self) -> LineString { - let mut combined_geometry = LineString::new(vec![]); - for leg in &self.legs { - combined_geometry.0.extend(&leg.geometry.0); - } - combined_geometry - } - pub fn from_valhalla(valhalla: &valhalla_api::Trip, mode: TravelMode) -> Self { let bounds = Rect::new( geo::coord!(x: valhalla.summary.min_lon, y: valhalla.summary.min_lat), @@ -173,11 +136,11 @@ impl Itinerary { let Some(first_leg) = legs_iter.next() else { return Err(Error::server("itinerary had no legs")); }; - let Some(mut itinerary_bounds) = first_leg.bounding_rect() else { + let Ok(Some(mut itinerary_bounds)) = first_leg.bounding_rect() else { return Err(Error::server("first leg has no bounding_rect")); }; for leg in legs_iter { - let Some(leg_bounds) = leg.bounding_rect() else { + let Ok(Some(leg_bounds)) = leg.bounding_rect() else { return Err(Error::server("leg has no bounding_rect")); }; extend_bounds(&mut itinerary_bounds, &leg_bounds); @@ -188,8 +151,8 @@ impl Itinerary { start_time: system_time_from_millis(itinerary.start_time), end_time: system_time_from_millis(itinerary.end_time), mode, - distance: convert_from_meters(distance_meters, distance_unit), - distance_units: distance_unit, + distance: distance_meters / 1000.0, + distance_units: DistanceUnit::Kilometers, bounds: itinerary_bounds, legs, }) @@ -222,18 +185,17 @@ impl From for Place { } } -#[derive(Debug, Serialize, Clone, PartialEq)] +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] #[serde(rename_all = "camelCase")] -pub(crate) struct Leg { +struct Leg { /// encoded polyline. 1e-6 scale, (lat, lon) - #[serde(serialize_with = "serialize_line_string_as_polyline6")] - geometry: LineString, + geometry: String, /// Which mode is this leg of the journey? - pub(crate) mode: TravelMode, + mode: TravelMode, #[serde(flatten)] - pub(crate) mode_leg: ModeLeg, + mode_leg: ModeLeg, /// Beginning of the leg from_place: Place, @@ -252,20 +214,14 @@ pub(crate) struct Leg { /// Start time of the leg #[serde(serialize_with = "serialize_system_time_as_millis")] end_time: SystemTime, - - /// Length of this leg, in units of the `distance_unit` of the Itinerary - distance: f64, - - /// Duration of this leg - pub(crate) duration_seconds: f64, } // Should we just pass the entire OTP leg? type TransitLeg = otp_api::Leg; -#[derive(Debug, Serialize, Clone, PartialEq)] +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] #[serde(rename_all = "camelCase")] -pub(crate) enum ModeLeg { +enum ModeLeg { // REVIEW: rename? There is a boolean field for OTP called TransitLeg #[serde(rename = "transitLeg")] Transit(Box), @@ -274,13 +230,13 @@ pub(crate) enum ModeLeg { NonTransit(Box), } -#[derive(Debug, Serialize, Clone, PartialEq)] +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] #[serde(rename_all = "camelCase")] -pub struct NonTransitLeg { - pub(crate) maneuvers: Vec, +struct NonTransitLeg { + maneuvers: Vec, /// The substantial road names along the route - pub(crate) substantial_street_names: Vec, + substantial_street_names: Vec, } impl NonTransitLeg { @@ -326,21 +282,18 @@ impl NonTransitLeg { // Eventually we might want to coalesce this into something not valhalla specific // but for now we only use it for valhalla trips -#[derive(Debug, Serialize, Clone, PartialEq)] +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] #[serde(rename_all = "camelCase")] pub struct Maneuver { pub instruction: Option, // pub cost: f64, // pub begin_shape_index: u64, // pub end_shape_index: u64, - #[serde(skip_serializing)] - pub geometry: LineString, // pub highway: Option, /// In units of the `distance_unit` of the trip leg pub distance: f64, pub street_names: Option>, - #[serde(skip_serializing)] - pub duration_seconds: f64, + // pub time: f64, // pub travel_mode: String, // pub travel_type: String, pub r#type: ManeuverType, @@ -352,30 +305,23 @@ pub struct Maneuver { impl Maneuver { fn from_valhalla(valhalla: valhalla_api::Maneuver, leg_geometry: &LineString) -> Self { - let coords = leg_geometry.0 - [valhalla.begin_shape_index as usize..=valhalla.end_shape_index as usize] - .to_owned(); - let geometry = LineString::from(coords); Self { instruction: Some(valhalla.instruction), street_names: valhalla.street_names, - duration_seconds: valhalla.time, r#type: valhalla.r#type, start_point: Point(leg_geometry[valhalla.begin_shape_index as usize]).into(), verbal_post_transition_instruction: valhalla.verbal_post_transition_instruction, distance: valhalla.length, - geometry, } } fn from_otp( otp: otp_api::Step, - geometry: LineString, - leg: &otp_api::Leg, + mode: otp_api::TransitMode, distance_unit: DistanceUnit, ) -> Self { let instruction = build_instruction( - leg.mode, + mode, otp.relative_direction, otp.absolute_direction, &otp.street_name, @@ -389,23 +335,20 @@ impl Maneuver { } else { Some(vec![otp.street_name]) }; - - let duration_seconds = otp.distance / leg.distance * leg.duration_seconds(); + let localized_distance = match distance_unit { + DistanceUnit::Kilometers => otp.distance, + // round to the nearest ten-thousandth + DistanceUnit::Miles => (otp.distance * 0.621371 * 10_000.0).round() / 10_000.0, + }; Self { instruction, r#type: otp.relative_direction.into(), street_names, verbal_post_transition_instruction, - distance: convert_from_meters(otp.distance, distance_unit), - duration_seconds, + distance: localized_distance, start_point: Point::new(otp.lon, otp.lat).into(), - geometry, } } - - pub(crate) fn distance_meters(&self, distance_unit: DistanceUnit) -> f64 { - convert_to_meters(self.distance, distance_unit) - } } // We could do so much better. Look at Valhalla's Odin. @@ -479,12 +422,13 @@ impl Leg { const VALHALLA_GEOMETRY_PRECISION: u32 = 6; const OTP_GEOMETRY_PRECISION: u32 = 5; - fn bounding_rect(&self) -> Option { - self.geometry.bounding_rect() + fn decoded_geometry(&self) -> std::result::Result { + decode_polyline(&self.geometry, Self::GEOMETRY_PRECISION) } - pub(crate) fn distance_meters(&self, itinerary_units: DistanceUnit) -> f64 { - convert_to_meters(self.distance, itinerary_units) + fn bounding_rect(&self) -> std::result::Result, String> { + let line_string = self.decoded_geometry()?; + Ok(line_string.bounding_rect()) } fn from_otp( @@ -493,12 +437,11 @@ impl Leg { distance_unit: DistanceUnit, ) -> std::result::Result { debug_assert_ne!(Self::OTP_GEOMETRY_PRECISION, Self::GEOMETRY_PRECISION); - let geometry = decode_polyline(&otp.leg_geometry.points, Self::OTP_GEOMETRY_PRECISION)?; + let line = decode_polyline(&otp.leg_geometry.points, Self::OTP_GEOMETRY_PRECISION)?; + let geometry = polyline::encode_coordinates(line, Self::GEOMETRY_PRECISION)?; let from_place: Place = (&otp.from).into(); let to_place: Place = (&otp.to).into(); - let mut distance_so_far = 0.0; - let mut segmenter = HaversineSegmenter::new(geometry.clone()); let mode_leg = match otp.mode { otp_api::TransitMode::Walk | otp_api::TransitMode::Bicycle @@ -507,13 +450,7 @@ impl Leg { .steps .iter() .cloned() - .map(|otp_step| { - // compute step geometry by distance along leg geometry - let step_geometry = - segmenter.next_segment(otp_step.distance).expect("TODO"); - distance_so_far += otp_step.distance; - Maneuver::from_otp(otp_step, step_geometry, otp, distance_unit) - }) + .map(|otp_step| Maneuver::from_otp(otp_step, otp.mode, distance_unit)) .collect(); // OTP doesn't include an arrival step like valhalla, so we synthesize one @@ -522,11 +459,9 @@ impl Leg { instruction: Some("Arrive at your destination.".to_string()), distance: 0.0, street_names: None, - duration_seconds: 0.0, r#type: ManeuverType::Destination, verbal_post_transition_instruction: None, start_point: to_place.location, - geometry: LineString::new(vec![to_place.location.into()]), }; maneuvers.push(maneuver); } @@ -546,8 +481,6 @@ impl Leg { end_time: system_time_from_millis(otp.end_time), geometry, mode: otp.mode.into(), - distance: convert_from_meters(otp.distance, distance_unit), - duration_seconds: otp.duration_seconds(), mode_leg, }) } @@ -561,8 +494,7 @@ impl Leg { to_place: LonLat, ) -> Self { let geometry = - polyline::decode_polyline(&valhalla.shape, Self::VALHALLA_GEOMETRY_PRECISION) - .expect("valid polyline from valhalla"); + polyline::decode_polyline(&valhalla.shape, Self::VALHALLA_GEOMETRY_PRECISION).unwrap(); let maneuvers = valhalla .maneuvers .iter() @@ -570,20 +502,53 @@ impl Leg { .map(|valhalla_maneuver| Maneuver::from_valhalla(valhalla_maneuver, &geometry)) .collect(); let leg = NonTransitLeg::new(maneuvers); + debug_assert_eq!(Self::VALHALLA_GEOMETRY_PRECISION, Self::GEOMETRY_PRECISION); Self { start_time, end_time, from_place: from_place.into(), to_place: to_place.into(), - geometry, + geometry: valhalla.shape.clone(), mode: travel_mode, mode_leg: ModeLeg::NonTransit(Box::new(leg)), - distance: valhalla.summary.length, - duration_seconds: valhalla.summary.time, } } } +// Comma separated list of travel modes +#[derive(Debug, Serialize, PartialEq, Eq, Clone)] +struct TravelModes(Vec); + +impl<'de> Deserialize<'de> for TravelModes { + fn deserialize(deserializer: D) -> std::result::Result + where + D: Deserializer<'de>, + { + struct CommaSeparatedVecVisitor; + + impl<'de> Visitor<'de> for CommaSeparatedVecVisitor { + type Value = TravelModes; + + fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { + formatter.write_str("a comma-separated string") + } + + fn visit_str(self, value: &str) -> std::result::Result + where + E: de::Error, + { + let modes = value + .split(',') + .map(|s| TravelMode::deserialize(s.into_deserializer())) + .collect::>()?; + Ok(TravelModes(modes)) + } + } + + deserializer.deserialize_str(CommaSeparatedVecVisitor) + } +} + impl actix_web::Responder for PlanResponseOk { type Body = actix_web::body::BoxBody; @@ -600,15 +565,7 @@ pub async fn get_plan( req: HttpRequest, app_state: web::Data, ) -> std::result::Result { - _get_plan(query, req, app_state).await -} - -pub async fn _get_plan( - query: web::Query, - req: HttpRequest, - app_state: web::Data, -) -> std::result::Result { - let Some(primary_mode) = query.mode.first() else { + let Some(primary_mode) = query.mode.0.first() else { return Err(PlanResponseErr::from(Error::user("mode is required"))); }; @@ -621,7 +578,7 @@ pub async fn _get_plan( match primary_mode { TravelMode::Transit => otp_plan(&query, req, &app_state, primary_mode).await, other => { - debug_assert!(query.mode.len() == 1, "valhalla only supports one mode"); + debug_assert!(query.mode.0.len() == 1, "valhalla only supports one mode"); let mode = match other { TravelMode::Transit => unreachable!("handled above"), @@ -796,8 +753,9 @@ mod tests { // legs assert_eq!(first_itinerary.legs.len(), 1); let first_leg = &first_itinerary.legs[0]; + let geometry = decode_polyline(&first_leg.geometry, 6).unwrap(); assert_relative_eq!( - first_leg.geometry.0[0], + geometry.0[0], geo::coord!(x: -122.33922, y: 47.57583), epsilon = 1e-4 ); @@ -835,14 +793,15 @@ mod tests { // itineraries let first_itinerary = &itineraries[0]; assert_eq!(first_itinerary.mode, TravelMode::Transit); - assert_relative_eq!(first_itinerary.distance, 6.311692992158277); + assert_relative_eq!(first_itinerary.distance, 10.157660000000002); assert_relative_eq!(first_itinerary.duration, 2347.0); // legs assert_eq!(first_itinerary.legs.len(), 4); let first_leg = &first_itinerary.legs[0]; + let geometry = polyline::decode_polyline(&first_leg.geometry, 6).unwrap(); assert_relative_eq!( - first_leg.geometry.0[0], + geometry.0[0], geo::coord!(x: -122.33922, y: 47.57583), epsilon = 1e-4 ); @@ -939,7 +898,7 @@ mod tests { .unwrap(); let first_maneuver = maneuvers.first().unwrap(); let expected_maneuver = json!({ - "distance": 0.0118992879068438, // TODO: truncate precision in serializer + "distance": 11.8993, "instruction": "Walk south on East Marginal Way South.", "startPoint": { "lat": 47.5758346, @@ -1047,7 +1006,7 @@ mod tests { { "begin_shape_index": 0, "cost": 246.056, - "end_shape_index": 1, + "end_shape_index": 69, "highway": true, "instruction": "Drive northeast on Fauntleroy Way Southwest.", "length": 2.218, diff --git a/services/travelmux/src/api/v5/directions.rs b/services/travelmux/src/api/v6/directions.rs similarity index 98% rename from services/travelmux/src/api/v5/directions.rs rename to services/travelmux/src/api/v6/directions.rs index b52368619..870589ca2 100644 --- a/services/travelmux/src/api/v5/directions.rs +++ b/services/travelmux/src/api/v6/directions.rs @@ -45,9 +45,9 @@ impl From for DirectionsResponseOk { #[cfg(test)] mod tests { + use super::osrm_api; + use super::PlanResponseOk; use super::*; - use crate::api::v5::error::PlanResponseOk; - use crate::api::v5::osrm_api; use crate::otp::otp_api; use crate::valhalla::valhalla_api; use crate::{DistanceUnit, TravelMode}; diff --git a/services/travelmux/src/api/v6/error.rs b/services/travelmux/src/api/v6/error.rs new file mode 100644 index 000000000..8e33623e8 --- /dev/null +++ b/services/travelmux/src/api/v6/error.rs @@ -0,0 +1,208 @@ +use crate::otp::otp_api; +use crate::valhalla::valhalla_api; +use crate::{DistanceUnit, Error, TravelMode}; +use actix_web::HttpResponseBuilder; +use serde::Serialize; + +use super::{Itinerary, Plan}; +use crate::error::ErrorType; +use actix_web::body::BoxBody; +use actix_web::HttpResponse; +use std::fmt; + +#[derive(Debug, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct UnboxedPlanResponseErr { + pub error: PlanError, + // The raw response from the upstream OTP /plan service + #[serde(rename = "_otp")] + _otp: Option, + + // The raw response from the upstream Valhalla /route service + #[serde(rename = "_valhalla")] + _valhalla: Option, +} +pub type PlanResponseErr = Box; + +#[derive(Debug, Serialize)] +#[serde(rename_all = "camelCase")] +pub struct PlanError { + pub status_code: u16, + pub error_code: u32, + pub message: String, +} + +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +pub struct PlanResponseOk { + pub(crate) plan: Plan, + + // The raw response from the upstream OTP /plan service + #[serde(rename = "_otp")] + _otp: Option, + + // The raw response from the upstream Valhalla /route service + #[serde(rename = "_valhalla")] + _valhalla: Option, +} + +impl From for PlanResponseErr { + fn from(value: valhalla_api::RouteResponseError) -> Self { + Self::new(UnboxedPlanResponseErr { + error: (&value).into(), + _valhalla: Some(value), + _otp: None, + }) + } +} + +impl From for PlanResponseErr { + fn from(value: otp_api::PlanError) -> Self { + Self::new(UnboxedPlanResponseErr { + error: (&value).into(), + _valhalla: None, + _otp: Some(value), + }) + } +} + +impl From<&valhalla_api::RouteResponseError> for PlanError { + fn from(value: &valhalla_api::RouteResponseError) -> Self { + PlanError { + status_code: value.status_code, + error_code: value.error_code + 2000, + message: value.error.clone(), + } + } +} + +impl fmt::Display for PlanResponseErr { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + write!( + f, + "status_code: {}, error_code: {}, message: {}", + self.error.status_code, self.error.error_code, self.error.message + ) + } +} + +impl std::error::Error for PlanResponseErr {} + +impl From for PlanResponseErr { + fn from(value: Error) -> Self { + Self::new(UnboxedPlanResponseErr { + error: value.into(), + _valhalla: None, + _otp: None, + }) + } +} + +impl From for PlanError { + fn from(value: Error) -> Self { + let error_code = value.error_type as u32; + match value.error_type { + ErrorType::NoCoverageForArea => Self { + status_code: 400, + error_code, + message: value.source.to_string(), + }, + ErrorType::User => Self { + status_code: 400, + error_code, + message: value.source.to_string(), + }, + ErrorType::Server => Self { + status_code: 500, + error_code, + message: value.source.to_string(), + }, + } + } +} + +impl From<&otp_api::PlanError> for PlanError { + fn from(value: &otp_api::PlanError) -> Self { + Self { + // This might be overzealous, but anecdotally, I haven't encountered any 500ish + // errors with OTP surfaced in this way yet + status_code: 400, + error_code: value.id, + message: value.msg.clone(), + } + } +} + +impl actix_web::ResponseError for PlanResponseErr { + fn status_code(&self) -> actix_web::http::StatusCode { + self.error.status_code.try_into().unwrap_or_else(|e| { + log::error!( + "invalid status code: {}, err: {e:?}", + self.error.status_code + ); + actix_web::http::StatusCode::INTERNAL_SERVER_ERROR + }) + } + + fn error_response(&self) -> HttpResponse { + HttpResponseBuilder::new(self.status_code()) + .content_type("application/json") + .json(self) + } +} + +impl PlanResponseOk { + pub fn from_otp( + mode: TravelMode, + mut otp: otp_api::PlanResponse, + distance_unit: DistanceUnit, + ) -> Result { + if let Some(otp_error) = otp.error { + return Err(otp_error.into()); + } + + otp.plan + .itineraries + .sort_by(|a, b| a.end_time.cmp(&b.end_time)); + + let itineraries_result: crate::Result> = otp + .plan + .itineraries + .iter() + .map(|itinerary: &otp_api::Itinerary| { + Itinerary::from_otp(itinerary, mode, distance_unit) + }) + .collect(); + + let itineraries = itineraries_result?; + + Ok(PlanResponseOk { + plan: Plan { itineraries }, + _otp: Some(otp), + _valhalla: None, + }) + } + + pub fn from_valhalla( + mode: TravelMode, + valhalla: valhalla_api::ValhallaRouteResponseResult, + ) -> Result { + let valhalla = match valhalla { + valhalla_api::ValhallaRouteResponseResult::Ok(valhalla) => valhalla, + valhalla_api::ValhallaRouteResponseResult::Err(err) => return Err(err), + }; + + let mut itineraries = vec![Itinerary::from_valhalla(&valhalla.trip, mode)]; + if let Some(alternates) = &valhalla.alternates { + for alternate in alternates { + itineraries.push(Itinerary::from_valhalla(&alternate.trip, mode)); + } + } + + Ok(PlanResponseOk { + plan: Plan { itineraries }, + _otp: None, + _valhalla: Some(valhalla), + }) + } +} diff --git a/services/travelmux/src/api/v6/mod.rs b/services/travelmux/src/api/v6/mod.rs new file mode 100644 index 000000000..45cd1f61b --- /dev/null +++ b/services/travelmux/src/api/v6/mod.rs @@ -0,0 +1,9 @@ +pub mod directions; +mod error; +mod osrm_api; +pub mod plan; +mod travel_modes; + +pub use travel_modes::TravelModes; + +pub use plan::{Itinerary, Plan}; diff --git a/services/travelmux/src/api/v5/osrm_api.rs b/services/travelmux/src/api/v6/osrm_api.rs similarity index 100% rename from services/travelmux/src/api/v5/osrm_api.rs rename to services/travelmux/src/api/v6/osrm_api.rs diff --git a/services/travelmux/src/api/v6/plan.rs b/services/travelmux/src/api/v6/plan.rs new file mode 100644 index 000000000..59f7fbac2 --- /dev/null +++ b/services/travelmux/src/api/v6/plan.rs @@ -0,0 +1,1110 @@ +use actix_web::{get, web, HttpRequest, HttpResponseBuilder}; +use geo::algorithm::BoundingRect; +use geo::geometry::{LineString, Point, Rect}; +use polyline::decode_polyline; +use reqwest::header::{HeaderName, HeaderValue}; +use serde::{Deserialize, Serialize}; +use std::collections::HashMap; +use std::time::{Duration, SystemTime}; + +use super::error::{PlanResponseErr, PlanResponseOk}; +use super::TravelModes; + +use crate::api::AppState; +use crate::error::ErrorType; +use crate::otp::otp_api; +use crate::otp::otp_api::{AbsoluteDirection, RelativeDirection}; +use crate::util::format::format_meters; +use crate::util::haversine_segmenter::HaversineSegmenter; +use crate::util::{ + deserialize_point_from_lat_lon, extend_bounds, serialize_line_string_as_polyline6, + serialize_rect_to_lng_lat, serialize_system_time_as_millis, system_time_from_millis, +}; +use crate::valhalla::valhalla_api; +use crate::valhalla::valhalla_api::{LonLat, ManeuverType}; +use crate::{DistanceUnit, Error, TravelMode}; + +const METERS_PER_MILE: f64 = 1609.34; + +fn convert_from_meters(meters: f64, output_units: DistanceUnit) -> f64 { + match output_units { + DistanceUnit::Kilometers => meters / 1000.0, + DistanceUnit::Miles => meters / METERS_PER_MILE, + } +} + +fn convert_to_meters(distance: f64, input_units: DistanceUnit) -> f64 { + match input_units { + DistanceUnit::Kilometers => distance * 1000.0, + DistanceUnit::Miles => distance * METERS_PER_MILE, + } +} + +#[derive(Debug, Deserialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +pub struct PlanQuery { + #[serde(deserialize_with = "deserialize_point_from_lat_lon")] + to_place: Point, + + #[serde(deserialize_with = "deserialize_point_from_lat_lon")] + from_place: Point, + + num_itineraries: u32, + + mode: TravelModes, + + /// Ignored by OTP - transit trips will always be metric. + /// Examine the `distance_units` in the response `Itinerary` to correctly interpret the response. + preferred_distance_units: Option, +} + +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +pub struct Plan { + pub(crate) itineraries: Vec, +} + +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +pub struct Itinerary { + mode: TravelMode, + /// seconds + pub(crate) duration: f64, + /// unix millis, UTC + #[serde(serialize_with = "serialize_system_time_as_millis")] + start_time: SystemTime, + /// unix millis, UTC + #[serde(serialize_with = "serialize_system_time_as_millis")] + end_time: SystemTime, + /// Units are in `distance_units` + distance: f64, + /// FIXME: I think we're returning meters even though distance unit is "Kilometers" + /// Probably we should rename DistanceUnit::Kilometers to DistanceUnit::Meters + /// This is passed as a parameter though, so it'd be a breaking change. + pub(crate) distance_units: DistanceUnit, + #[serde(serialize_with = "serialize_rect_to_lng_lat")] + bounds: Rect, + pub(crate) legs: Vec, +} + +impl Itinerary { + pub fn distance_meters(&self) -> f64 { + match self.distance_units { + DistanceUnit::Kilometers => self.distance * 1000.0, + DistanceUnit::Miles => self.distance * METERS_PER_MILE, + } + } + + pub fn combined_geometry(&self) -> LineString { + let mut combined_geometry = LineString::new(vec![]); + for leg in &self.legs { + combined_geometry.0.extend(&leg.geometry.0); + } + combined_geometry + } + + pub fn from_valhalla(valhalla: &valhalla_api::Trip, mode: TravelMode) -> Self { + let bounds = Rect::new( + geo::coord!(x: valhalla.summary.min_lon, y: valhalla.summary.min_lat), + geo::coord!(x: valhalla.summary.max_lon, y: valhalla.summary.max_lat), + ); + + let start_time = SystemTime::now(); + let end_time = start_time + Duration::from_millis((valhalla.summary.time * 1000.0) as u64); + debug_assert!( + valhalla.locations.len() == valhalla.legs.len() + 1, + "assuming each leg has a start and end location" + ); + + let mut start_time = SystemTime::now(); + let legs = valhalla + .legs + .iter() + .zip(valhalla.locations.windows(2)) + .map(|(v_leg, locations)| { + let leg_start_time = start_time; + let leg_end_time = + start_time + Duration::from_millis((v_leg.summary.time * 1000.0) as u64); + start_time = leg_end_time; + Leg::from_valhalla( + v_leg, + mode, + leg_start_time, + leg_end_time, + locations[0], + locations[1], + ) + }) + .collect(); + + Self { + mode, + start_time, + end_time, + duration: valhalla.summary.time, + distance: valhalla.summary.length, + bounds, + distance_units: valhalla.units, + legs, + } + } + + pub fn from_otp( + itinerary: &otp_api::Itinerary, + mode: TravelMode, + distance_unit: DistanceUnit, + ) -> crate::Result { + // OTP responses are always in meters + let distance_meters: f64 = itinerary.legs.iter().map(|l| l.distance).sum(); + let Ok(legs): std::result::Result, _> = itinerary + .legs + .iter() + .enumerate() + .map(|(idx, leg)| { + let is_destination_leg = idx == itinerary.legs.len() - 1; + Leg::from_otp(leg, is_destination_leg, distance_unit) + }) + .collect() + else { + return Err(Error::server("failed to parse legs")); + }; + + let mut legs_iter = legs.iter(); + let Some(first_leg) = legs_iter.next() else { + return Err(Error::server("itinerary had no legs")); + }; + let Some(mut itinerary_bounds) = first_leg.bounding_rect() else { + return Err(Error::server("first leg has no bounding_rect")); + }; + for leg in legs_iter { + let Some(leg_bounds) = leg.bounding_rect() else { + return Err(Error::server("leg has no bounding_rect")); + }; + extend_bounds(&mut itinerary_bounds, &leg_bounds); + } + + Ok(Self { + duration: itinerary.duration as f64, + start_time: system_time_from_millis(itinerary.start_time), + end_time: system_time_from_millis(itinerary.end_time), + mode, + distance: convert_from_meters(distance_meters, distance_unit), + distance_units: distance_unit, + bounds: itinerary_bounds, + legs, + }) + } +} + +#[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +struct Place { + #[serde(flatten)] + location: LonLat, + name: Option, +} + +impl From<&otp_api::Place> for Place { + fn from(value: &otp_api::Place) -> Self { + Self { + location: value.location.into(), + name: value.name.clone(), + } + } +} + +impl From for Place { + fn from(value: LonLat) -> Self { + Self { + location: value, + name: None, + } + } +} + +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +pub(crate) struct Leg { + /// encoded polyline. 1e-6 scale, (lat, lon) + #[serde(serialize_with = "serialize_line_string_as_polyline6")] + geometry: LineString, + + /// Which mode is this leg of the journey? + pub(crate) mode: TravelMode, + + #[serde(flatten)] + pub(crate) mode_leg: ModeLeg, + + /// Beginning of the leg + from_place: Place, + + /// End of the Leg + to_place: Place, + + // This is mostly OTP specific. We can synthesize a value from the valhalla response, but we + // don't currently use it. + /// Start time of the leg + #[serde(serialize_with = "serialize_system_time_as_millis")] + start_time: SystemTime, + + // This is mostly OTP specific. We can synthesize a value from the valhalla response, but we + // don't currently use it. + /// Start time of the leg + #[serde(serialize_with = "serialize_system_time_as_millis")] + end_time: SystemTime, + + /// Length of this leg, in units of the `distance_unit` of the Itinerary + distance: f64, + + /// Duration of this leg + pub(crate) duration_seconds: f64, +} + +// Should we just pass the entire OTP leg? +type TransitLeg = otp_api::Leg; + +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +pub(crate) enum ModeLeg { + // REVIEW: rename? There is a boolean field for OTP called TransitLeg + #[serde(rename = "transitLeg")] + Transit(Box), + + #[serde(rename = "nonTransitLeg")] + NonTransit(Box), +} + +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +pub struct NonTransitLeg { + pub(crate) maneuvers: Vec, + + /// The substantial road names along the route + pub(crate) substantial_street_names: Vec, +} + +impl NonTransitLeg { + fn new(maneuvers: Vec) -> Self { + let mut street_distances = HashMap::new(); + for maneuver in &maneuvers { + if let Some(street_names) = &maneuver.street_names { + for street_name in street_names { + *street_distances.entry(street_name).or_insert(0.0) += maneuver.distance; + } + } + } + let mut scores: Vec<_> = street_distances.into_iter().collect(); + scores.sort_by(|(_, a), (_, b)| b.partial_cmp(a).unwrap_or(std::cmp::Ordering::Equal)); + + let limit = 3; + // Don't include tiny segments in the description of the route + let mut inclusion_threshold = None; + + let substantial_street_names = scores + .into_iter() + .take(limit) + .flat_map(|(street_name, distance)| { + let Some(inclusion_threshold) = inclusion_threshold else { + // don't consider streets that are much smaller than this one + inclusion_threshold = Some(distance * 0.5); + return Some(street_name.clone()); + }; + if distance > inclusion_threshold { + Some(street_name.clone()) + } else { + None + } + }) + .collect(); + + Self { + maneuvers, + substantial_street_names, + } + } +} + +// Eventually we might want to coalesce this into something not valhalla specific +// but for now we only use it for valhalla trips +#[derive(Debug, Serialize, Clone, PartialEq)] +#[serde(rename_all = "camelCase")] +pub struct Maneuver { + pub instruction: Option, + // pub cost: f64, + // pub begin_shape_index: u64, + // pub end_shape_index: u64, + #[serde(skip_serializing)] + pub geometry: LineString, + // pub highway: Option, + /// In units of the `distance_unit` of the trip leg + pub distance: f64, + pub street_names: Option>, + #[serde(skip_serializing)] + pub duration_seconds: f64, + // pub travel_mode: String, + // pub travel_type: String, + pub r#type: ManeuverType, + pub verbal_post_transition_instruction: Option, + pub start_point: LonLat, + // pub verbal_pre_transition_instruction: Option, + // pub verbal_succinct_transition_instruction: Option, +} + +impl Maneuver { + fn from_valhalla(valhalla: valhalla_api::Maneuver, leg_geometry: &LineString) -> Self { + let coords = leg_geometry.0 + [valhalla.begin_shape_index as usize..=valhalla.end_shape_index as usize] + .to_owned(); + let geometry = LineString::from(coords); + Self { + instruction: Some(valhalla.instruction), + street_names: valhalla.street_names, + duration_seconds: valhalla.time, + r#type: valhalla.r#type, + start_point: Point(leg_geometry[valhalla.begin_shape_index as usize]).into(), + verbal_post_transition_instruction: valhalla.verbal_post_transition_instruction, + distance: valhalla.length, + geometry, + } + } + + fn from_otp( + otp: otp_api::Step, + geometry: LineString, + leg: &otp_api::Leg, + distance_unit: DistanceUnit, + ) -> Self { + let instruction = build_instruction( + leg.mode, + otp.relative_direction, + otp.absolute_direction, + &otp.street_name, + ); + + let verbal_post_transition_instruction = + build_verbal_post_transition_instruction(otp.distance, distance_unit); + + let street_names = if let Some(true) = otp.bogus_name { + None + } else { + Some(vec![otp.street_name]) + }; + + let duration_seconds = otp.distance / leg.distance * leg.duration_seconds(); + Self { + instruction, + r#type: otp.relative_direction.into(), + street_names, + verbal_post_transition_instruction, + distance: convert_from_meters(otp.distance, distance_unit), + duration_seconds, + start_point: Point::new(otp.lon, otp.lat).into(), + geometry, + } + } + + pub(crate) fn distance_meters(&self, distance_unit: DistanceUnit) -> f64 { + convert_to_meters(self.distance, distance_unit) + } +} + +// We could do so much better. Look at Valhalla's Odin. +// +// e.g. take context of previous maneuver. "Bear right to stay on Main Street" +fn build_instruction( + mode: otp_api::TransitMode, + maneuver_type: otp_api::RelativeDirection, + absolute_direction: Option, + street_name: &str, +) -> Option { + match maneuver_type { + RelativeDirection::Depart => { + if let Some(absolute_direction) = absolute_direction { + let direction = match absolute_direction { + AbsoluteDirection::North => "north", + AbsoluteDirection::Northeast => "northeast", + AbsoluteDirection::East => "east", + AbsoluteDirection::Southeast => "southeast", + AbsoluteDirection::South => "south", + AbsoluteDirection::Southwest => "southwest", + AbsoluteDirection::West => "west", + AbsoluteDirection::Northwest => "northwest", + }; + let mode = match mode { + otp_api::TransitMode::Walk => "Walk", + otp_api::TransitMode::Bicycle => "Bike", + otp_api::TransitMode::Car => "Drive", + _ => "Transit", + }; + Some(format!("{mode} {direction} on {street_name}.")) + } else { + Some("Depart.".to_string()) + } + } + RelativeDirection::HardLeft => Some(format!("Turn left onto {street_name}.")), + RelativeDirection::Left => Some(format!("Turn left onto {street_name}.")), + RelativeDirection::SlightlyLeft => Some(format!("Turn slightly left onto {street_name}.")), + RelativeDirection::Continue => Some(format!("Continue onto {street_name}.")), + RelativeDirection::SlightlyRight => { + Some(format!("Turn slightly right onto {street_name}.")) + } + RelativeDirection::Right => Some(format!("Turn right onto {street_name}.")), + RelativeDirection::HardRight => Some(format!("Turn right onto {street_name}.")), + RelativeDirection::CircleClockwise | RelativeDirection::CircleCounterclockwise => { + Some("Enter the roundabout.".to_string()) + } + RelativeDirection::Elevator => Some("Enter the elevator.".to_string()), + RelativeDirection::UturnLeft | RelativeDirection::UturnRight => { + Some("Make a U-turn.".to_string()) + } + } +} + +fn build_verbal_post_transition_instruction( + distance: f64, + distance_unit: DistanceUnit, +) -> Option { + if distance == 0.0 { + None + } else { + Some(format!( + "Continue for {}.", + format_meters(distance, distance_unit) + )) + } +} + +impl Leg { + const GEOMETRY_PRECISION: u32 = 6; + const VALHALLA_GEOMETRY_PRECISION: u32 = 6; + const OTP_GEOMETRY_PRECISION: u32 = 5; + + fn bounding_rect(&self) -> Option { + self.geometry.bounding_rect() + } + + pub(crate) fn distance_meters(&self, itinerary_units: DistanceUnit) -> f64 { + convert_to_meters(self.distance, itinerary_units) + } + + fn from_otp( + otp: &otp_api::Leg, + is_destination_leg: bool, + distance_unit: DistanceUnit, + ) -> std::result::Result { + debug_assert_ne!(Self::OTP_GEOMETRY_PRECISION, Self::GEOMETRY_PRECISION); + let geometry = decode_polyline(&otp.leg_geometry.points, Self::OTP_GEOMETRY_PRECISION)?; + let from_place: Place = (&otp.from).into(); + let to_place: Place = (&otp.to).into(); + + let mut distance_so_far = 0.0; + let mut segmenter = HaversineSegmenter::new(geometry.clone()); + let mode_leg = match otp.mode { + otp_api::TransitMode::Walk + | otp_api::TransitMode::Bicycle + | otp_api::TransitMode::Car => { + let mut maneuvers: Vec<_> = otp + .steps + .iter() + .cloned() + .map(|otp_step| { + // compute step geometry by distance along leg geometry + let step_geometry = + segmenter.next_segment(otp_step.distance).expect("TODO"); + distance_so_far += otp_step.distance; + Maneuver::from_otp(otp_step, step_geometry, otp, distance_unit) + }) + .collect(); + + // OTP doesn't include an arrival step like valhalla, so we synthesize one + if is_destination_leg { + let maneuver = Maneuver { + instruction: Some("Arrive at your destination.".to_string()), + distance: 0.0, + street_names: None, + duration_seconds: 0.0, + r#type: ManeuverType::Destination, + verbal_post_transition_instruction: None, + start_point: to_place.location, + geometry: LineString::new(vec![to_place.location.into()]), + }; + maneuvers.push(maneuver); + } + let leg = NonTransitLeg::new(maneuvers); + ModeLeg::NonTransit(Box::new(leg)) + } + _ => { + // assume everything else is transit + ModeLeg::Transit(Box::new(otp.clone())) + } + }; + + Ok(Self { + from_place, + to_place, + start_time: system_time_from_millis(otp.start_time), + end_time: system_time_from_millis(otp.end_time), + geometry, + mode: otp.mode.into(), + distance: convert_from_meters(otp.distance, distance_unit), + duration_seconds: otp.duration_seconds(), + mode_leg, + }) + } + + fn from_valhalla( + valhalla: &valhalla_api::Leg, + travel_mode: TravelMode, + start_time: SystemTime, + end_time: SystemTime, + from_place: LonLat, + to_place: LonLat, + ) -> Self { + let geometry = + polyline::decode_polyline(&valhalla.shape, Self::VALHALLA_GEOMETRY_PRECISION) + .expect("valid polyline from valhalla"); + let maneuvers = valhalla + .maneuvers + .iter() + .cloned() + .map(|valhalla_maneuver| Maneuver::from_valhalla(valhalla_maneuver, &geometry)) + .collect(); + let leg = NonTransitLeg::new(maneuvers); + Self { + start_time, + end_time, + from_place: from_place.into(), + to_place: to_place.into(), + geometry, + mode: travel_mode, + mode_leg: ModeLeg::NonTransit(Box::new(leg)), + distance: valhalla.summary.length, + duration_seconds: valhalla.summary.time, + } + } +} + +impl actix_web::Responder for PlanResponseOk { + type Body = actix_web::body::BoxBody; + + fn respond_to(self, _req: &HttpRequest) -> actix_web::HttpResponse { + let mut response = HttpResponseBuilder::new(actix_web::http::StatusCode::OK); + response.content_type("application/json"); + response.json(self) + } +} + +#[get("/v5/plan")] +pub async fn get_plan( + query: web::Query, + req: HttpRequest, + app_state: web::Data, +) -> std::result::Result { + _get_plan(query, req, app_state).await +} + +pub async fn _get_plan( + query: web::Query, + req: HttpRequest, + app_state: web::Data, +) -> std::result::Result { + let Some(primary_mode) = query.mode.first() else { + return Err(PlanResponseErr::from(Error::user("mode is required"))); + }; + + let distance_units = query + .preferred_distance_units + .unwrap_or(DistanceUnit::Kilometers); + + // TODO: Handle bus+bike if bike is first, for now all our clients are responsible for enforcing that + // the "primary" mode appears first. + match primary_mode { + TravelMode::Transit => otp_plan(&query, req, &app_state, primary_mode).await, + other => { + debug_assert!(query.mode.len() == 1, "valhalla only supports one mode"); + + let mode = match other { + TravelMode::Transit => unreachable!("handled above"), + TravelMode::Bicycle => valhalla_api::ModeCosting::Bicycle, + TravelMode::Car => valhalla_api::ModeCosting::Auto, + TravelMode::Walk => valhalla_api::ModeCosting::Pedestrian, + }; + + // route?json={%22locations%22:[{%22lat%22:47.575837,%22lon%22:-122.339414},{%22lat%22:47.651048,%22lon%22:-122.347234}],%22costing%22:%22auto%22,%22alternates%22:3,%22units%22:%22miles%22} + let router_url = app_state.valhalla_router().plan_url( + query.from_place, + query.to_place, + mode, + query.num_itineraries, + distance_units, + )?; + let valhalla_response: reqwest::Response = + reqwest::get(router_url).await.map_err(|e| { + log::error!("error while fetching from valhalla service: {e}"); + PlanResponseErr::from(Error::server(e)) + })?; + if !valhalla_response.status().is_success() { + log::warn!( + "upstream HTTP Error from valhalla service: {}", + valhalla_response.status() + ) + } + + let mut response = HttpResponseBuilder::new(valhalla_response.status()); + debug_assert_eq!( + valhalla_response + .headers() + .get(HeaderName::from_static("content-type")), + Some(&HeaderValue::from_str("application/json;charset=utf-8").unwrap()) + ); + response.content_type("application/json;charset=utf-8"); + + let valhalla_route_response: valhalla_api::ValhallaRouteResponseResult = + valhalla_response.json().await.map_err(|e| { + log::error!("error while parsing valhalla response: {e}"); + PlanResponseErr::from(Error::server(e)) + })?; + + let mut plan_response = + PlanResponseOk::from_valhalla(*primary_mode, valhalla_route_response)?; + + if primary_mode == &TravelMode::Bicycle || primary_mode == &TravelMode::Walk { + match otp_plan(&query, req, &app_state, primary_mode).await { + Ok(mut otp_response) => { + debug_assert_eq!( + 1, + otp_response.plan.itineraries.len(), + "expected exactly one itinerary from OTP" + ); + if let Some(otp_itinerary) = otp_response.plan.itineraries.pop() { + log::debug!("adding OTP itinerary to valhalla response"); + plan_response.plan.itineraries.insert(0, otp_itinerary); + } + } + Err(e) => { + // match error_code to raw value of ErrorType enum + match ErrorType::try_from(e.error.error_code) { + Ok(ErrorType::NoCoverageForArea) => { + log::debug!("No OTP coverage for route"); + } + other => { + debug_assert!(other.is_ok(), "unexpected error code: {e:?}"); + // We're mixing with results from Valhalla anyway, so dont' surface this error + // to the user. Likely we just don't support this area. + log::error!("OTP failed to plan {primary_mode:?} route: {e}"); + } + } + } + } + } + + Ok(plan_response) + } + } +} + +async fn otp_plan( + query: &web::Query, + req: HttpRequest, + app_state: &web::Data, + primary_mode: &TravelMode, +) -> Result { + let Some(mut router_url) = app_state + .otp_cluster() + .find_router_url(query.from_place, query.to_place) + else { + Err( + Error::user("Transit directions not available for this area.") + .error_type(ErrorType::NoCoverageForArea), + )? + }; + + // if we end up building this manually rather than passing it through, we'll need to be sure + // to handle the bike+bus case + router_url.set_query(Some(req.query_string())); + log::debug!( + "found matching router. Forwarding request to: {}", + router_url + ); + + let otp_response: reqwest::Response = reqwest::get(router_url).await.map_err(|e| { + log::error!("error while fetching from otp service: {e}"); + PlanResponseErr::from(Error::server(e)) + })?; + if !otp_response.status().is_success() { + log::warn!( + "upstream HTTP Error from otp service: {}", + otp_response.status() + ) + } + + let mut response = HttpResponseBuilder::new(otp_response.status()); + debug_assert_eq!( + otp_response + .headers() + .get(HeaderName::from_static("content-type")), + Some(&HeaderValue::from_str("application/json").unwrap()) + ); + response.content_type("application/json"); + + let otp_plan_response: otp_api::PlanResponse = otp_response.json().await.map_err(|e| { + log::error!("error while parsing otp response: {e}"); + PlanResponseErr::from(Error::server(e)) + })?; + PlanResponseOk::from_otp( + *primary_mode, + otp_plan_response, + query + .preferred_distance_units + .unwrap_or(DistanceUnit::Kilometers), + ) +} + +#[cfg(test)] +mod tests { + use super::*; + use approx::assert_relative_eq; + use serde_json::{json, Value}; + use std::fs::File; + use std::io::BufReader; + + #[test] + fn parse_from_valhalla() { + let stubbed_response = + File::open("tests/fixtures/requests/valhalla_pedestrian_route.json").unwrap(); + let valhalla: valhalla_api::RouteResponse = + serde_json::from_reader(BufReader::new(stubbed_response)).unwrap(); + + let valhalla_response_result = valhalla_api::ValhallaRouteResponseResult::Ok(valhalla); + let plan_response = + PlanResponseOk::from_valhalla(TravelMode::Walk, valhalla_response_result).unwrap(); + assert_eq!(plan_response.plan.itineraries.len(), 3); + + // itineraries + let first_itinerary = &plan_response.plan.itineraries[0]; + assert_eq!(first_itinerary.mode, TravelMode::Walk); + assert_relative_eq!(first_itinerary.distance, 5.684); + assert_relative_eq!(first_itinerary.duration, 6488.443); + assert_relative_eq!( + first_itinerary.bounds, + Rect::new( + geo::coord!(x: -122.347201, y: 47.575663), + geo::coord!(x: -122.335618, y: 47.651047) + ) + ); + + // legs + assert_eq!(first_itinerary.legs.len(), 1); + let first_leg = &first_itinerary.legs[0]; + assert_relative_eq!( + first_leg.geometry.0[0], + geo::coord!(x: -122.33922, y: 47.57583), + epsilon = 1e-4 + ); + + assert_relative_eq!( + geo::Point::from(first_leg.from_place.location), + geo::point!(x: -122.339414, y: 47.575837) + ); + assert_relative_eq!( + geo::Point::from(first_leg.to_place.location), + geo::point!(x:-122.347234, y: 47.651048) + ); + assert!(first_leg.to_place.name.is_none()); + + let ModeLeg::NonTransit(non_transit_leg) = &first_leg.mode_leg else { + panic!("unexpected non-transit leg") + }; + + assert_eq!(first_leg.mode, TravelMode::Walk); + assert_eq!(non_transit_leg.maneuvers.len(), 21); + } + + #[test] + fn parse_from_otp() { + let stubbed_response = + File::open("tests/fixtures/requests/opentripplanner_transit_plan.json").unwrap(); + let otp: otp_api::PlanResponse = + serde_json::from_reader(BufReader::new(stubbed_response)).unwrap(); + let plan_response = + PlanResponseOk::from_otp(TravelMode::Transit, otp, DistanceUnit::Miles).unwrap(); + + let itineraries = plan_response.plan.itineraries; + assert_eq!(itineraries.len(), 6); + + // itineraries + let first_itinerary = &itineraries[0]; + assert_eq!(first_itinerary.mode, TravelMode::Transit); + assert_relative_eq!(first_itinerary.distance, 6.311692992158277); + assert_relative_eq!(first_itinerary.duration, 2347.0); + + // legs + assert_eq!(first_itinerary.legs.len(), 4); + let first_leg = &first_itinerary.legs[0]; + assert_relative_eq!( + first_leg.geometry.0[0], + geo::coord!(x: -122.33922, y: 47.57583), + epsilon = 1e-4 + ); + + assert_relative_eq!( + geo::Point::from(first_leg.from_place.location), + geo::point!(x: -122.339414, y: 47.575837) + ); + assert_relative_eq!( + geo::Point::from(first_leg.to_place.location), + geo::point!(x: -122.334106, y: 47.575924) + ); + assert_eq!( + first_leg.to_place.name.as_ref().unwrap(), + "1st Ave S & S Hanford St" + ); + + assert_eq!(first_leg.mode, TravelMode::Walk); + let ModeLeg::NonTransit(non_transit_leg) = &first_leg.mode_leg else { + panic!("expected non-transit leg") + }; + let maneuvers = &non_transit_leg.maneuvers; + assert_eq!(maneuvers.len(), 2); + assert_eq!(maneuvers[0].r#type, ManeuverType::Start); + assert_eq!(maneuvers[1].r#type, ManeuverType::Left); + + let transit_leg = &first_itinerary.legs[2]; + assert_eq!(transit_leg.mode, TravelMode::Transit); + let ModeLeg::Transit(transit_leg) = &transit_leg.mode_leg else { + panic!("expected transit leg") + }; + assert!(transit_leg.route_color.is_none()); + } + + #[test] + fn serialize_response_from_otp() { + let stubbed_response = + File::open("tests/fixtures/requests/opentripplanner_transit_plan.json").unwrap(); + let otp: otp_api::PlanResponse = + serde_json::from_reader(BufReader::new(stubbed_response)).unwrap(); + let plan_response = + PlanResponseOk::from_otp(TravelMode::Transit, otp, DistanceUnit::Miles).unwrap(); + let response = serde_json::to_string(&plan_response).unwrap(); + let parsed_response: serde_json::Value = serde_json::from_str(&response).unwrap(); + let response_object = parsed_response.as_object().expect("expected Object"); + let plan = response_object + .get("plan") + .unwrap() + .as_object() + .expect("expected Object"); + let first_itinerary = plan + .get("itineraries") + .unwrap() + .as_array() + .unwrap() + .first() + .unwrap(); + let legs = first_itinerary.get("legs").unwrap().as_array().unwrap(); + + // Verify walking leg + let first_leg = legs.first().unwrap().as_object().unwrap(); + let mode = first_leg.get("mode").unwrap().as_str().unwrap(); + assert_eq!(mode, "WALK"); + + let mode = first_leg + .get("startTime") + .expect("field missing") + .as_u64() + .expect("unexpected type. expected u64"); + assert_eq!(mode, 1715974501000); + + let mode = first_leg + .get("endTime") + .expect("field missing") + .as_u64() + .expect("unexpected type. expected u64"); + assert_eq!(mode, 1715974870000); + + assert!(first_leg.get("transitLeg").is_none()); + let non_transit_leg = first_leg.get("nonTransitLeg").unwrap().as_object().unwrap(); + + let substantial_street_names = non_transit_leg + .get("substantialStreetNames") + .unwrap() + .as_array() + .unwrap(); + let expected_names = vec!["East Marginal Way South"]; + assert_eq!(substantial_street_names, &expected_names); + + let maneuvers = non_transit_leg + .get("maneuvers") + .unwrap() + .as_array() + .unwrap(); + let first_maneuver = maneuvers.first().unwrap(); + let expected_maneuver = json!({ + "distance": 0.0118992879068438, // TODO: truncate precision in serializer + "instruction": "Walk south on East Marginal Way South.", + "startPoint": { + "lat": 47.5758346, + "lon": -122.3392181 + }, + "streetNames": ["East Marginal Way South"], + "type": 1, + "verbalPostTransitionInstruction": "Continue for 60 feet." + }); + assert_eq!(first_maneuver, &expected_maneuver); + + // Verify Transit leg + let transit_leg = legs.get(2).unwrap().as_object().unwrap(); + let mode = transit_leg.get("mode").unwrap().as_str().unwrap(); + assert_eq!(mode, "TRANSIT"); + assert!(transit_leg.get("maneuvers").is_none()); + let transit_leg = transit_leg + .get("transitLeg") + .unwrap() + .as_object() + .expect("json object"); + + // Brittle: If the fixtures are updated, these values might change due to time of day or whatever. + assert_eq!( + transit_leg.get("agencyName").unwrap().as_str().unwrap(), + "Metro Transit" + ); + + assert!(transit_leg.get("route").is_none()); + } + + #[test] + fn serialize_response_from_valhalla() { + let stubbed_response = + File::open("tests/fixtures/requests/valhalla_pedestrian_route.json").unwrap(); + let valhalla: valhalla_api::RouteResponse = + serde_json::from_reader(BufReader::new(stubbed_response)).unwrap(); + + let valhalla_response_result = valhalla_api::ValhallaRouteResponseResult::Ok(valhalla); + let plan_response = + PlanResponseOk::from_valhalla(TravelMode::Walk, valhalla_response_result).unwrap(); + + let response = serde_json::to_string(&plan_response).unwrap(); + let parsed_response: serde_json::Value = serde_json::from_str(&response).unwrap(); + let response_object = parsed_response.as_object().expect("expected Object"); + let plan = response_object + .get("plan") + .unwrap() + .as_object() + .expect("expected Object"); + let first_itinerary = plan + .get("itineraries") + .unwrap() + .as_array() + .unwrap() + .first() + .unwrap(); + let legs = first_itinerary.get("legs").unwrap().as_array().unwrap(); + + // Verify walking leg + let first_leg = legs.first().unwrap().as_object().unwrap(); + let mode = first_leg.get("mode").unwrap().as_str().unwrap(); + assert_eq!(mode, "WALK"); + assert!(first_leg.get("transitLeg").is_none()); + let non_transit_leg = first_leg.get("nonTransitLeg").unwrap().as_object().unwrap(); + + let substantial_street_names = non_transit_leg + .get("substantialStreetNames") + .unwrap() + .as_array() + .unwrap(); + assert_eq!( + substantial_street_names, + &[ + "Dexter Avenue", + "East Marginal Way South", + "Alaskan Way South" + ] + ); + + let maneuvers = non_transit_leg + .get("maneuvers") + .unwrap() + .as_array() + .unwrap(); + let first_maneuver = maneuvers.first().unwrap(); + let expected_maneuver = json!({ + "type": 2, + "instruction": "Walk south on East Marginal Way South.", + "verbalPostTransitionInstruction": "Continue for 60 feet.", + "distance": 0.011, + "startPoint": { + "lat": 47.575836, + "lon": -122.339216 + }, + "streetNames": ["East Marginal Way South"], + }); + assert_eq!(first_maneuver, &expected_maneuver); + } + + #[test] + fn parse_maneuver_from_valhalla_json() { + // deserialize a maneuver from a JSON string + let json = r#" + { + "begin_shape_index": 0, + "cost": 246.056, + "end_shape_index": 1, + "highway": true, + "instruction": "Drive northeast on Fauntleroy Way Southwest.", + "length": 2.218, + "street_names": [ + "Fauntleroy Way Southwest" + ], + "time": 198.858, + "travel_mode": "drive", + "travel_type": "car", + "type": 2, + "verbal_post_transition_instruction": "Continue for 2 miles.", + "verbal_pre_transition_instruction": "Drive northeast on Fauntleroy Way Southwest.", + "verbal_succinct_transition_instruction": "Drive northeast." + }"#; + + let valhalla_maneuver: valhalla_api::Maneuver = serde_json::from_str(json).unwrap(); + assert_eq!(valhalla_maneuver.r#type, ManeuverType::StartRight); + assert_eq!( + valhalla_maneuver.instruction, + "Drive northeast on Fauntleroy Way Southwest." + ); + + // fake geometry + let geometry = LineString::from(vec![ + geo::coord!(x: -122.398, y: 47.564), + geo::coord!(x: -122.396, y: 47.566), + ]); + let maneuver = Maneuver::from_valhalla(valhalla_maneuver, &geometry); + let actual = serde_json::to_string(&maneuver).unwrap(); + // parse the JSON string back into an Object Value + let actual_object: Value = serde_json::from_str(&actual).unwrap(); + + let expected_object = serde_json::json!({ + "instruction": "Drive northeast on Fauntleroy Way Southwest.", + "type": 2, + "distance": 2.218, + "startPoint": { "lon": -122.398, "lat": 47.564}, + "streetNames": ["Fauntleroy Way Southwest"], + "verbalPostTransitionInstruction": "Continue for 2 miles.", + }); + + assert_eq!(actual_object, expected_object); + } + + #[test] + fn parse_error_from_valhalla() { + let json = serde_json::json!({ + "error_code": 154, + "error": "Path distance exceeds the max distance limit: 200000 meters", + "status_code": 400, + "status": "Bad Request" + }) + .to_string(); + + let valhalla_error: valhalla_api::RouteResponseError = serde_json::from_str(&json).unwrap(); + let plan_error = PlanResponseErr::from(valhalla_error); + assert_eq!(plan_error.error.status_code, 400); + assert_eq!(plan_error.error.error_code, 2154); + } +} diff --git a/services/travelmux/src/api/v5/travel_modes.rs b/services/travelmux/src/api/v6/travel_modes.rs similarity index 100% rename from services/travelmux/src/api/v5/travel_modes.rs rename to services/travelmux/src/api/v6/travel_modes.rs diff --git a/services/travelmux/src/bin/travelmux-server/main.rs b/services/travelmux/src/bin/travelmux-server/main.rs index 44d50bb69..be44f5f70 100644 --- a/services/travelmux/src/bin/travelmux-server/main.rs +++ b/services/travelmux/src/bin/travelmux-server/main.rs @@ -50,7 +50,8 @@ async fn main() -> Result<()> { .app_data(web::Data::new(app_state.clone())) .service(api::v4::plan::get_plan) .service(api::v5::plan::get_plan) - .service(api::v5::directions::get_directions) + .service(api::v6::plan::get_plan) + .service(api::v6::directions::get_directions) .service(api::health::get_ready) .service(api::health::get_alive) }) diff --git a/services/travelmux/src/api/v5/haversine_segmenter.rs b/services/travelmux/src/util/haversine_segmenter.rs similarity index 97% rename from services/travelmux/src/api/v5/haversine_segmenter.rs rename to services/travelmux/src/util/haversine_segmenter.rs index d66d99fc2..996cd87e2 100644 --- a/services/travelmux/src/api/v5/haversine_segmenter.rs +++ b/services/travelmux/src/util/haversine_segmenter.rs @@ -58,7 +58,7 @@ impl HaversineSegmenter { mod test { use super::*; use approx::assert_relative_eq; - use geo::{point, wkt, HaversineDestination}; + use geo::{point, HaversineLength}; #[test] fn test_segmenter() { @@ -83,7 +83,7 @@ mod test { // next one should pick up where the last one left off let segment_2 = segmenter.next_segment(10_000.0).unwrap(); - assert_eq!(segment_1.0.last(), segment_2.0.first()); + assert_eq!(segment_1.0.last().unwrap(), segment_2.0.first().unwrap()); let east_of_berlin = point!(x: 13.482210264987538, y: 52.34640526357316); let segment_3 = segmenter.next_segment(paris_to_berlin_distance).unwrap(); diff --git a/services/travelmux/src/util/mod.rs b/services/travelmux/src/util/mod.rs index 53d2a6536..fb4928bcf 100644 --- a/services/travelmux/src/util/mod.rs +++ b/services/travelmux/src/util/mod.rs @@ -1,4 +1,5 @@ pub mod format; +pub mod haversine_segmenter; use geo::{Point, Rect}; use serde::{