diff --git a/crates/shared/src/price_estimation/factory.rs b/crates/shared/src/price_estimation/factory.rs index 2d374eca7e..351f450483 100644 --- a/crates/shared/src/price_estimation/factory.rs +++ b/crates/shared/src/price_estimation/factory.rs @@ -559,12 +559,7 @@ impl PriceEstimatorCreating for HttpPriceEstimator { impl From<&LegacySolver> for HttpPriceEstimatorParams { fn from(solver: &LegacySolver) -> Self { - let base = { - let mut url = solver.url.clone(); - url.set_path(""); - url - }; - let solve_path = solver.url.path().to_owned(); + let (base, solve_path) = crate::url::split_at_path(&solver.url).unwrap(); Self { base, solve_path, diff --git a/crates/shared/src/url.rs b/crates/shared/src/url.rs index c310838e01..40d3d4312c 100644 --- a/crates/shared/src/url.rs +++ b/crates/shared/src/url.rs @@ -1,4 +1,7 @@ -use url::Url; +use { + anyhow::{Context, Result}, + url::Url, +}; /// Join a path with a URL, ensuring that there is only one slash between them. /// It doesn't matter if the URL ends with a slash or the path starts with one. @@ -12,3 +15,66 @@ pub fn join(url: &Url, mut path: &str) -> Url { } Url::parse(&format!("{url}/{path}")).unwrap() } + +/// Splits a URL right where the path begins into base and endpoint. +/// https://my.solver.xyz/solve/1?param=1#fragment=some +/// becomes +/// (https://my.solver.xyz/, /solve/1?param=1#fragment=some) +/// Path that were split like this can be joined to the original URL using +/// [`join`]. +pub fn split_at_path(url: &Url) -> Result<(Url, String)> { + let base = format!( + "{}://{}{}/", + url.scheme(), + url.host().context("URL should have a host")?, + url.port() + .map(|port| format!(":{port}")) + .unwrap_or_default() + ) + .parse() + .expect("stripping off the path is always safe"); + let endpoint = format!( + "{}{}{}", + url.path(), + url.query() + .map(|params| format!("?{params}")) + .unwrap_or_default(), + url.fragment() + .map(|params| format!("#{params}")) + .unwrap_or_default(), + ); + Ok((base, endpoint)) +} + +#[cfg(test)] +mod tests { + use super::*; + + /// Tests that we can split a URL and join it back together without messing + /// up the URL on the way. + #[test] + fn split_and_join() { + let round_trip = |s: &str| { + let url = s.parse().unwrap(); + let (base, endpoint) = split_at_path(&url).unwrap(); + assert_eq!(url, join(&base, &endpoint)); + }; + + // base + port + path + multiple params + multiple fragments + round_trip("https://my.solver.xyz:1234/solve/1?param1=1¶m2=2#fragment=1&fragment2=2"); + // base + path + multiple params + multiple fragments + round_trip("https://my.solver.xyz/solve/1?param1=1¶m2=2#fragment=1&fragment2=2"); + // base + path + multiple params + round_trip("https://my.solver.xyz/solve/1?param1=1¶m2=2"); + // base + path + multiple fragments + round_trip("https://my.solver.xyz/solve/1#fragment1=1&fragment2=2"); + // base + path + single param + single fragment + round_trip("https://my.solver.xyz/solve/1?param=1#fragment1=1"); + // base + path + round_trip("https://my.solver.xyz/solve/1"); + // base + round_trip("http://my.solver.xyz"); + // base + multiple params + multiple fragments + round_trip("https://my.solver.xyz?param1=1¶m2=2#fragment=1&fragment2=2"); + } +} diff --git a/crates/solvers/src/boundary/legacy.rs b/crates/solvers/src/boundary/legacy.rs index 30388eed37..97ad158069 100644 --- a/crates/solvers/src/boundary/legacy.rs +++ b/crates/solvers/src/boundary/legacy.rs @@ -44,9 +44,7 @@ pub struct Legacy { impl Legacy { pub fn new(config: crate::domain::solver::legacy::Config) -> Self { - let solve_path = config.endpoint.path().to_owned(); - let mut base = config.endpoint; - base.set_path(""); + let (base, solve_path) = shared::url::split_at_path(&config.endpoint).unwrap(); Self { solver: DefaultHttpSolverApi {