diff --git a/url/src/lib.rs b/url/src/lib.rs index 0be004e6f..bd4272cc5 100644 --- a/url/src/lib.rs +++ b/url/src/lib.rs @@ -678,12 +678,22 @@ impl Url { ); } else { // Anarchist URL (no authority) - assert_eq!(self.username_end, self.scheme_end + 1); - assert_eq!(self.host_start, self.scheme_end + 1); - assert_eq!(self.host_end, self.scheme_end + 1); assert_eq!(self.host, HostInternal::None); assert_eq!(self.port, None); - assert_eq!(self.path_start, self.scheme_end + 1); + if self.path().starts_with("//") { + // special case when first path fragment is empty + assert_eq!(self.byte_at(self.scheme_end + 1), b'/'); + assert_eq!(self.byte_at(self.scheme_end + 2), b'.'); + assert_eq!(self.username_end, self.scheme_end + 3); + assert_eq!(self.host_start, self.scheme_end + 3); + assert_eq!(self.host_end, self.scheme_end + 3); + assert_eq!(self.path_start, self.scheme_end + 3); + } else { + assert_eq!(self.username_end, self.scheme_end + 1); + assert_eq!(self.host_start, self.scheme_end + 1); + assert_eq!(self.host_end, self.scheme_end + 1); + assert_eq!(self.path_start, self.scheme_end + 1); + } } if let Some(start) = self.query_start { assert!(start >= self.path_start); diff --git a/url/src/parser.rs b/url/src/parser.rs index 7829e03a0..70b0aa2c4 100644 --- a/url/src/parser.rs +++ b/url/src/parser.rs @@ -467,19 +467,29 @@ impl<'a> Parser<'a> { return self.after_double_slash(input, scheme_type, scheme_end); } // Anarchist URL (no authority) - let path_start = to_u32(self.serialization.len())?; - let username_end = path_start; - let host_start = path_start; - let host_end = path_start; - let host = HostInternal::None; - let port = None; + let mut path_start = self.serialization.len(); let remaining = if let Some(input) = input.split_prefix('/') { - let path_start = self.serialization.len(); self.serialization.push('/'); self.parse_path(scheme_type, &mut false, path_start, input) } else { self.parse_cannot_be_a_base_path(input) }; + // This prevents web+demo:/.//not-a-host/ or web+demo:/path/..//not-a-host/, + // when parsed and then serialized, from ending up as web+demo://not-a-host/ + // (they end up as web+demo:/.//not-a-host/). + if self.serialization[path_start..].starts_with("//") { + // If url’s host is null, url does not have an opaque path, + // url’s path’s size is greater than 1, and url’s path[0] is the empty string, + // then append U+002F (/) followed by U+002E (.) to output. + self.serialization.insert_str(path_start, "/."); + path_start += 2; + } + let path_start = to_u32(path_start)?; + let username_end = path_start; + let host_start = path_start; + let host_end = path_start; + let host = HostInternal::None; + let port = None; self.with_query_and_fragment( scheme_type, scheme_end, @@ -1281,16 +1291,6 @@ impl<'a> Parser<'a> { self.serialization.push_str(path.trim_start_matches('/')); } - // This prevents web+demo:/.//not-a-host/ or web+demo:/path/..//not-a-host/, - // when parsed and then serialized, from ending up as web+demo://not-a-host/ - // (they end up as web+demo:/.//not-a-host/). - if !*has_host && self.serialization[path_start..].starts_with("//") { - // If url’s host is null, url does not have an opaque path, - // url’s path’s size is greater than 1, and url’s path[0] is the empty string, - // then append U+002F (/) followed by U+002E (.) to output. - self.serialization.insert_str(path_start, "/."); - } - input } diff --git a/url/tests/urltestdata.json b/url/tests/urltestdata.json index 8bda080a9..1a352bff1 100644 --- a/url/tests/urltestdata.json +++ b/url/tests/urltestdata.json @@ -7487,7 +7487,6 @@ "hash": "" }, "Serialize /. in path", - "skip next", { "input": "non-spec:/.//", "base": "about:blank", @@ -7502,7 +7501,6 @@ "search": "", "hash": "" }, - "skip next", { "input": "non-spec:/..//", "base": "about:blank", @@ -7517,7 +7515,6 @@ "search": "", "hash": "" }, - "skip next", { "input": "non-spec:/a/..//", "base": "about:blank", @@ -7532,7 +7529,6 @@ "search": "", "hash": "" }, - "skip next", { "input": "non-spec:/.//path", "base": "about:blank", @@ -7547,7 +7543,6 @@ "search": "", "hash": "" }, - "skip next", { "input": "non-spec:/..//path", "base": "about:blank", @@ -7562,7 +7557,6 @@ "search": "", "hash": "" }, - "skip next", { "input": "non-spec:/a/..//path", "base": "about:blank", @@ -7637,7 +7631,6 @@ "search": "", "hash": "" }, - "skip next", { "input": "", "base": "non-spec:/..//p", @@ -7652,7 +7645,6 @@ "search": "", "hash": "" }, - "skip next", { "input": "path", "base": "non-spec:/..//p",