2021-12-09 16:00:47 -05:00
|
|
|
// Loose subset interpretation of the URL standard
|
|
|
|
// Not using full Url crate here for no_std compatibility
|
|
|
|
//
|
|
|
|
// Caveats:
|
|
|
|
// No support for query string parsing
|
|
|
|
// No support for paths with ';' parameters
|
|
|
|
// URLs must convert to UTF8
|
|
|
|
// Only IP address and DNS hostname host fields are supported
|
|
|
|
|
|
|
|
use super::IpAddr;
|
2021-12-17 19:18:25 -05:00
|
|
|
use alloc::borrow::ToOwned;
|
|
|
|
use alloc::string::String;
|
|
|
|
use alloc::vec::Vec;
|
2021-12-09 16:00:47 -05:00
|
|
|
use core::fmt;
|
|
|
|
use core::str::FromStr;
|
|
|
|
|
|
|
|
fn is_alphanum(c: u8) -> bool {
|
|
|
|
matches!(c,
|
|
|
|
b'A'..=b'Z'
|
|
|
|
| b'a'..=b'z'
|
|
|
|
| b'0'..=b'9'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
fn is_mark(c: u8) -> bool {
|
|
|
|
matches!(
|
|
|
|
c,
|
|
|
|
b'-' | b'_' | b'.' | b'!' | b'~' | b'*' | b'\'' | b'(' | b')'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
fn is_unreserved(c: u8) -> bool {
|
|
|
|
is_alphanum(c) || is_mark(c)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn must_encode_userinfo(c: u8) -> bool {
|
|
|
|
!(is_unreserved(c) || matches!(c, b'%' | b':' | b';' | b'&' | b'=' | b'+' | b'$' | b','))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn must_encode_path(c: u8) -> bool {
|
|
|
|
!(is_unreserved(c)
|
|
|
|
|| matches!(
|
|
|
|
c,
|
|
|
|
b'%' | b'/' | b':' | b'@' | b'&' | b'=' | b'+' | b'$' | b','
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn is_valid_host<H: AsRef<str>>(host: H) -> bool {
|
|
|
|
if host.as_ref().is_empty() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if IpAddr::from_str(host.as_ref()).is_err() {
|
|
|
|
for ch in host.as_ref().chars() {
|
|
|
|
if !matches!(ch,
|
|
|
|
'A'..='Z' | 'a'..='z' | '0'..='9' | '-' | '.' )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
|
|
|
fn is_valid_scheme<H: AsRef<str>>(host: H) -> bool {
|
|
|
|
let mut chars = host.as_ref().chars();
|
|
|
|
if let Some(ch) = chars.next() {
|
|
|
|
if !matches!(ch, 'A'..='Z' | 'a'..='z') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for ch in chars {
|
|
|
|
if !matches!(ch,
|
|
|
|
'A'..='Z' | 'a'..='z' | '0'..='9' | '-' | '+' | '.' )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
|
|
|
fn hex_decode(h: u8) -> Result<u8, String> {
|
|
|
|
match h {
|
|
|
|
b'0'..=b'9' => Ok(h - b'0'),
|
|
|
|
b'A'..=b'F' => Ok(h - b'A' + 10),
|
|
|
|
b'a'..=b'f' => Ok(h - b'a' + 10),
|
|
|
|
_ => Err("Unexpected character in percent encoding".to_owned()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn hex_encode(c: u8) -> (char, char) {
|
|
|
|
let c0 = c >> 4;
|
|
|
|
let c1 = c & 15;
|
|
|
|
(
|
|
|
|
if c0 < 10 {
|
|
|
|
char::from_u32((b'0' + c0) as u32).unwrap()
|
|
|
|
} else {
|
|
|
|
char::from_u32((b'A' + c0 - 10) as u32).unwrap()
|
|
|
|
},
|
|
|
|
if c1 < 10 {
|
|
|
|
char::from_u32((b'0' + c1) as u32).unwrap()
|
|
|
|
} else {
|
|
|
|
char::from_u32((b'A' + c1 - 10) as u32).unwrap()
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn url_decode<S: AsRef<str>>(s: S) -> Result<String, String> {
|
|
|
|
let url = s.as_ref().to_owned();
|
|
|
|
if !url.is_ascii() {
|
|
|
|
return Err("URL is not in ASCII encoding".to_owned());
|
|
|
|
}
|
|
|
|
let url_bytes = url.as_bytes();
|
|
|
|
let mut dec_bytes: Vec<u8> = Vec::with_capacity(url_bytes.len());
|
|
|
|
let mut i = 0;
|
|
|
|
let end = url_bytes.len();
|
|
|
|
while i < end {
|
|
|
|
let mut b = url_bytes[i];
|
|
|
|
i += 1;
|
|
|
|
if b == b'%' {
|
|
|
|
if (i + 1) >= end {
|
|
|
|
return Err("Invalid URL encoding".to_owned());
|
|
|
|
}
|
|
|
|
b = hex_decode(url_bytes[i])? << 4 | hex_decode(url_bytes[i + 1])?;
|
|
|
|
i += 2;
|
|
|
|
}
|
|
|
|
dec_bytes.push(b);
|
|
|
|
}
|
|
|
|
String::from_utf8(dec_bytes).map_err(|e| format!("Decoded URL is not valid UTF-8: {}", e))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn url_encode<S: AsRef<str>>(s: S, must_encode: impl Fn(u8) -> bool) -> String {
|
|
|
|
let bytes = s.as_ref().as_bytes();
|
|
|
|
let mut out = String::new();
|
|
|
|
for b in bytes {
|
|
|
|
if must_encode(*b) {
|
|
|
|
let (c0, c1) = hex_encode(*b);
|
|
|
|
out.push('%');
|
|
|
|
out.push(c0);
|
|
|
|
out.push(c1);
|
|
|
|
} else {
|
|
|
|
out.push(char::from_u32(*b as u32).unwrap())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out
|
|
|
|
}
|
|
|
|
|
|
|
|
fn convert_port<N>(port_str: N) -> Result<u16, String>
|
|
|
|
where
|
|
|
|
N: AsRef<str>,
|
|
|
|
{
|
|
|
|
port_str
|
|
|
|
.as_ref()
|
|
|
|
.parse::<u16>()
|
|
|
|
.map_err(|e| format!("Invalid port: {}", e))
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
|
|
|
pub struct SplitUrlPath {
|
|
|
|
pub path: String,
|
|
|
|
pub fragment: Option<String>,
|
|
|
|
pub query: Option<String>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SplitUrlPath {
|
|
|
|
pub fn new<P, F, Q>(path: P, fragment: Option<F>, query: Option<Q>) -> Self
|
|
|
|
where
|
|
|
|
P: AsRef<str>,
|
|
|
|
F: AsRef<str>,
|
|
|
|
Q: AsRef<str>,
|
|
|
|
{
|
|
|
|
Self {
|
|
|
|
path: path.as_ref().to_owned(),
|
|
|
|
fragment: fragment.map(|f| f.as_ref().to_owned()),
|
|
|
|
query: query.map(|f| f.as_ref().to_owned()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromStr for SplitUrlPath {
|
|
|
|
type Err = String;
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
|
Ok(if let Some((p, q)) = s.split_once('?') {
|
|
|
|
if let Some((p, f)) = p.split_once('#') {
|
|
|
|
SplitUrlPath::new(url_decode(p)?, Some(url_decode(f)?), Some(q))
|
|
|
|
} else {
|
|
|
|
SplitUrlPath::new(url_decode(p)?, Option::<String>::None, Some(q))
|
|
|
|
}
|
|
|
|
} else if let Some((p, f)) = s.split_once('#') {
|
|
|
|
SplitUrlPath::new(url_decode(p)?, Some(url_decode(f)?), Option::<String>::None)
|
|
|
|
} else {
|
|
|
|
SplitUrlPath::new(
|
|
|
|
url_decode(s)?,
|
|
|
|
Option::<String>::None,
|
|
|
|
Option::<String>::None,
|
|
|
|
)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for SplitUrlPath {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
if let Some(fragment) = &self.fragment {
|
|
|
|
if let Some(query) = &self.query {
|
|
|
|
write!(
|
|
|
|
f,
|
|
|
|
"{}#{}?{}",
|
|
|
|
url_encode(&self.path, must_encode_path),
|
|
|
|
url_encode(fragment, must_encode_path),
|
|
|
|
query
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
write!(f, "{}#{}", self.path, fragment)
|
|
|
|
}
|
|
|
|
} else if let Some(query) = &self.query {
|
|
|
|
write!(f, "{}?{}", url_encode(&self.path, must_encode_path), query)
|
|
|
|
} else {
|
|
|
|
write!(f, "{}", url_encode(&self.path, must_encode_path))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
|
|
|
pub struct SplitUrl {
|
|
|
|
pub scheme: String,
|
|
|
|
pub userinfo: Option<String>,
|
|
|
|
pub host: String,
|
|
|
|
pub port: Option<u16>,
|
|
|
|
pub path: Option<SplitUrlPath>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SplitUrl {
|
|
|
|
pub fn new<S, H>(
|
|
|
|
scheme: S,
|
|
|
|
userinfo: Option<String>,
|
|
|
|
host: H,
|
|
|
|
port: Option<u16>,
|
|
|
|
path: Option<SplitUrlPath>,
|
|
|
|
) -> Self
|
|
|
|
where
|
|
|
|
S: AsRef<str>,
|
|
|
|
H: AsRef<str>,
|
|
|
|
{
|
|
|
|
Self {
|
|
|
|
scheme: scheme.as_ref().to_owned(),
|
|
|
|
userinfo,
|
|
|
|
host: host.as_ref().to_owned(),
|
|
|
|
port,
|
|
|
|
path,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromStr for SplitUrl {
|
|
|
|
type Err = String;
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
|
if let Some((scheme, mut rest)) = s.split_once("://") {
|
|
|
|
if !is_valid_scheme(scheme) {
|
|
|
|
return Err("Invalid scheme specified".to_owned());
|
|
|
|
}
|
|
|
|
let userinfo = {
|
2022-03-13 12:45:36 -04:00
|
|
|
if let Some((userinfo_str, after)) = rest.split_once('@') {
|
2021-12-09 16:00:47 -05:00
|
|
|
rest = after;
|
|
|
|
Some(url_decode(userinfo_str)?)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
};
|
|
|
|
if let Some((host, rest)) = rest.rsplit_once(':') {
|
|
|
|
if !is_valid_host(host) {
|
|
|
|
return Err("Invalid host specified".to_owned());
|
|
|
|
}
|
|
|
|
if let Some((portstr, path)) = rest.split_once('/') {
|
|
|
|
let port = convert_port(portstr)?;
|
|
|
|
let path = SplitUrlPath::from_str(path)?;
|
|
|
|
Ok(SplitUrl::new(
|
|
|
|
scheme,
|
|
|
|
userinfo,
|
|
|
|
host,
|
|
|
|
Some(port),
|
|
|
|
Some(path),
|
|
|
|
))
|
|
|
|
} else {
|
|
|
|
let port = convert_port(rest)?;
|
|
|
|
Ok(SplitUrl::new(scheme, userinfo, host, Some(port), None))
|
|
|
|
}
|
|
|
|
} else if let Some((host, path)) = rest.split_once('/') {
|
|
|
|
if !is_valid_host(host) {
|
|
|
|
return Err("Invalid host specified".to_owned());
|
|
|
|
}
|
|
|
|
let path = SplitUrlPath::from_str(path)?;
|
|
|
|
Ok(SplitUrl::new(scheme, userinfo, host, None, Some(path)))
|
|
|
|
} else {
|
|
|
|
if !is_valid_host(rest) {
|
|
|
|
return Err("Invalid host specified".to_owned());
|
|
|
|
}
|
|
|
|
Ok(SplitUrl::new(scheme, userinfo, rest, None, None))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Err("No scheme specified".to_owned())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for SplitUrl {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
let hostname = {
|
|
|
|
if let Some(userinfo) = &self.userinfo {
|
|
|
|
let userinfo = url_encode(userinfo, must_encode_userinfo);
|
|
|
|
if let Some(port) = self.port {
|
|
|
|
format!("{}@{}:{}", userinfo, self.host, port)
|
|
|
|
} else {
|
|
|
|
format!("{}@{}", userinfo, self.host)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
self.host.clone()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
if let Some(path) = &self.path {
|
|
|
|
write!(f, "{}://{}/{}", self.scheme, hostname, path)
|
|
|
|
} else {
|
|
|
|
write!(f, "{}://{}", self.scheme, hostname)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|