@@ -1756,6 +1756,39 @@ impl Url {
1756
1756
let old_after_path_pos = to_u32 ( self . serialization . len ( ) ) . unwrap ( ) ;
1757
1757
let cannot_be_a_base = self . cannot_be_a_base ( ) ;
1758
1758
let scheme_type = SchemeType :: from ( self . scheme ( ) ) ;
1759
+ let mut path_empty = false ;
1760
+
1761
+ // Check ':' and then see if the next character is '/'
1762
+ let mut has_host = if let Some ( index) = self . serialization . find ( ":" ) {
1763
+ if self . serialization . len ( ) > index + 1
1764
+ && self . serialization . as_bytes ( ) . get ( index + 1 ) == Some ( & b'/' )
1765
+ {
1766
+ let rest = & self . serialization [ ( index + ":/" . len ( ) ) ..] ;
1767
+ let host_part = rest. split ( '/' ) . next ( ) . unwrap_or ( "" ) ;
1768
+ path_empty = rest. is_empty ( ) ;
1769
+ !host_part. is_empty ( ) && !host_part. contains ( '@' )
1770
+ } else {
1771
+ false
1772
+ }
1773
+ } else {
1774
+ false
1775
+ } ;
1776
+
1777
+ // Ensure the path length is greater than 1 to account
1778
+ // for cases where "/." is already appended from serialization
1779
+ // If we set path, then we already checked the other two conditions:
1780
+ // https://url.spec.whatwg.org/#url-serializing
1781
+ // 1. The host is null
1782
+ // 2. the first segment of the URL's path is an empty string
1783
+ if self . path ( ) . len ( ) + path. len ( ) > 1 {
1784
+ if let Some ( index) = self . serialization . find ( ":" ) {
1785
+ let removal_start = index + ":" . len ( ) ;
1786
+ if self . serialization [ removal_start..] . starts_with ( "/." ) {
1787
+ self . path_start = removal_start as u32 ;
1788
+ }
1789
+ }
1790
+ }
1791
+
1759
1792
self . serialization . truncate ( self . path_start as usize ) ;
1760
1793
self . mutate ( |parser| {
1761
1794
if cannot_be_a_base {
@@ -1765,14 +1798,33 @@ impl Url {
1765
1798
}
1766
1799
parser. parse_cannot_be_a_base_path ( parser:: Input :: new_no_trim ( path) ) ;
1767
1800
} else {
1768
- let mut has_host = true ; // FIXME
1769
1801
parser. parse_path_start (
1770
1802
scheme_type,
1771
1803
& mut has_host,
1772
1804
parser:: Input :: new_no_trim ( path) ,
1773
1805
) ;
1774
1806
}
1775
1807
} ) ;
1808
+
1809
+ // For cases where normalization is applied across both the serialization and the path.
1810
+ // Append "/." immediately after the scheme (up to ":")
1811
+ // This is done if three conditions are met.
1812
+ // https://url.spec.whatwg.org/#url-serializing
1813
+ // 1. The host is null
1814
+ // 2. The url's path length is greater than 1
1815
+ // 3. the first segment of the URL's path is an empty string
1816
+ if !has_host && path. len ( ) > 1 && path_empty {
1817
+ if let Some ( index) = self . serialization . find ( ":" ) {
1818
+ if self . serialization . len ( ) > index + 2
1819
+ && self . serialization . as_bytes ( ) . get ( index + 1 ) == Some ( & b'/' )
1820
+ && self . serialization . as_bytes ( ) . get ( index + 2 ) == Some ( & b'/' )
1821
+ {
1822
+ self . serialization . insert_str ( index + ":" . len ( ) , "/." ) ;
1823
+ self . path_start += "/." . len ( ) as u32 ;
1824
+ }
1825
+ }
1826
+ }
1827
+
1776
1828
self . restore_after_path ( old_after_path_pos, & after_path) ;
1777
1829
}
1778
1830
0 commit comments