rust/src/libstd/net_url.rs
2013-05-10 02:46:18 -04:00

1076 lines
31 KiB
Rust

// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
//! Types/fns concerning URLs (see RFC 3986)
use core::cmp::Eq;
use core::io::{Reader, ReaderUtil};
use core::io;
use core::hashmap::HashMap;
use core::str;
use core::to_bytes;
use core::uint;
#[deriving(Clone, Eq)]
struct Url {
scheme: ~str,
user: Option<UserInfo>,
host: ~str,
port: Option<~str>,
path: ~str,
query: Query,
fragment: Option<~str>
}
#[deriving(Clone, Eq)]
struct UserInfo {
user: ~str,
pass: Option<~str>
}
pub type Query = ~[(~str, ~str)];
pub impl Url {
fn new(
scheme: ~str,
user: Option<UserInfo>,
host: ~str,
port: Option<~str>,
path: ~str,
query: Query,
fragment: Option<~str>
) -> Url {
Url {
scheme: scheme,
user: user,
host: host,
port: port,
path: path,
query: query,
fragment: fragment,
}
}
}
pub impl UserInfo {
fn new(user: ~str, pass: Option<~str>) -> UserInfo {
UserInfo { user: user, pass: pass }
}
}
fn encode_inner(s: &str, full_url: bool) -> ~str {
do io::with_str_reader(s) |rdr| {
let mut out = ~"";
while !rdr.eof() {
let ch = rdr.read_byte() as char;
match ch {
// unreserved:
'A' .. 'Z' |
'a' .. 'z' |
'0' .. '9' |
'-' | '.' | '_' | '~' => {
str::push_char(&mut out, ch);
}
_ => {
if full_url {
match ch {
// gen-delims:
':' | '/' | '?' | '#' | '[' | ']' | '@' |
// sub-delims:
'!' | '$' | '&' | '"' | '(' | ')' | '*' |
'+' | ',' | ';' | '=' => {
str::push_char(&mut out, ch);
}
_ => out += fmt!("%%%X", ch as uint)
}
} else {
out += fmt!("%%%X", ch as uint);
}
}
}
}
out
}
}
/**
* Encodes a URI by replacing reserved characters with percent encoded
* character sequences.
*
* This function is compliant with RFC 3986.
*/
pub fn encode(s: &str) -> ~str {
encode_inner(s, true)
}
/**
* Encodes a URI component by replacing reserved characters with percent
* encoded character sequences.
*
* This function is compliant with RFC 3986.
*/
pub fn encode_component(s: &str) -> ~str {
encode_inner(s, false)
}
fn decode_inner(s: &str, full_url: bool) -> ~str {
do io::with_str_reader(s) |rdr| {
let mut out = ~"";
while !rdr.eof() {
match rdr.read_char() {
'%' => {
let bytes = rdr.read_bytes(2u);
let ch = uint::parse_bytes(bytes, 16u).get() as char;
if full_url {
// Only decode some characters:
match ch {
// gen-delims:
':' | '/' | '?' | '#' | '[' | ']' | '@' |
// sub-delims:
'!' | '$' | '&' | '"' | '(' | ')' | '*' |
'+' | ',' | ';' | '=' => {
str::push_char(&mut out, '%');
str::push_char(&mut out, bytes[0u] as char);
str::push_char(&mut out, bytes[1u] as char);
}
ch => str::push_char(&mut out, ch)
}
} else {
str::push_char(&mut out, ch);
}
}
ch => str::push_char(&mut out, ch)
}
}
out
}
}
/**
* Decode a string encoded with percent encoding.
*
* This will only decode escape sequences generated by encode.
*/
pub fn decode(s: &str) -> ~str {
decode_inner(s, true)
}
/**
* Decode a string encoded with percent encoding.
*/
pub fn decode_component(s: &str) -> ~str {
decode_inner(s, false)
}
fn encode_plus(s: &str) -> ~str {
do io::with_str_reader(s) |rdr| {
let mut out = ~"";
while !rdr.eof() {
let ch = rdr.read_byte() as char;
match ch {
'A' .. 'Z' | 'a' .. 'z' | '0' .. '9' | '_' | '.' | '-' => {
str::push_char(&mut out, ch);
}
' ' => str::push_char(&mut out, '+'),
_ => out += fmt!("%%%X", ch as uint)
}
}
out
}
}
/**
* Encode a hashmap to the 'application/x-www-form-urlencoded' media type.
*/
pub fn encode_form_urlencoded(m: &HashMap<~str, ~[~str]>) -> ~str {
let mut out = ~"";
let mut first = true;
for m.each |key, values| {
let key = encode_plus(*key);
for values.each |value| {
if first {
first = false;
} else {
str::push_char(&mut out, '&');
first = false;
}
out += fmt!("%s=%s", key, encode_plus(*value));
}
}
out
}
/**
* Decode a string encoded with the 'application/x-www-form-urlencoded' media
* type into a hashmap.
*/
pub fn decode_form_urlencoded(s: &[u8]) -> HashMap<~str, ~[~str]> {
do io::with_bytes_reader(s) |rdr| {
let mut m = HashMap::new();
let mut key = ~"";
let mut value = ~"";
let mut parsing_key = true;
while !rdr.eof() {
match rdr.read_char() {
'&' | ';' => {
if key != ~"" && value != ~"" {
let mut values = match m.pop(&key) {
Some(values) => values,
None => ~[],
};
values.push(value);
m.insert(key, values);
}
parsing_key = true;
key = ~"";
value = ~"";
}
'=' => parsing_key = false,
ch => {
let ch = match ch {
'%' => {
let bytes = rdr.read_bytes(2u);
uint::parse_bytes(bytes, 16u).get() as char
}
'+' => ' ',
ch => ch
};
if parsing_key {
str::push_char(&mut key, ch)
} else {
str::push_char(&mut value, ch)
}
}
}
}
if key != ~"" && value != ~"" {
let mut values = match m.pop(&key) {
Some(values) => values,
None => ~[],
};
values.push(value);
m.insert(key, values);
}
m
}
}
fn split_char_first(s: &str, c: char) -> (~str, ~str) {
let len = str::len(s);
let mut index = len;
let mut mat = 0;
do io::with_str_reader(s) |rdr| {
let mut ch;
while !rdr.eof() {
ch = rdr.read_byte() as char;
if ch == c {
// found a match, adjust markers
index = rdr.tell()-1;
mat = 1;
break;
}
}
}
if index+mat == len {
return (str::slice(s, 0, index).to_owned(), ~"");
} else {
return (str::slice(s, 0, index).to_owned(),
str::slice(s, index + mat, str::len(s)).to_owned());
}
}
fn userinfo_from_str(uinfo: &str) -> UserInfo {
let (user, p) = split_char_first(uinfo, ':');
let pass = if str::len(p) == 0 {
None
} else {
Some(p)
};
return UserInfo::new(user, pass);
}
fn userinfo_to_str(userinfo: &UserInfo) -> ~str {
match userinfo.pass {
Some(ref pass) => fmt!("%s:%s@", userinfo.user, *pass),
None => fmt!("%s@", userinfo.user),
}
}
fn query_from_str(rawquery: &str) -> Query {
let mut query: Query = ~[];
if str::len(rawquery) != 0 {
for str::each_split_char(rawquery, '&') |p| {
let (k, v) = split_char_first(p, '=');
query.push((decode_component(k), decode_component(v)));
};
}
return query;
}
pub fn query_to_str(query: &Query) -> ~str {
let mut strvec = ~[];
for query.each |kv| {
match kv {
&(ref k, ref v) => {
strvec.push(fmt!("%s=%s",
encode_component(*k),
encode_component(*v))
);
}
}
}
return str::connect(strvec, ~"&");
}
// returns the scheme and the rest of the url, or a parsing error
pub fn get_scheme(rawurl: &str) -> Result<(~str, ~str), ~str> {
for str::each_chari(rawurl) |i,c| {
match c {
'A' .. 'Z' | 'a' .. 'z' => loop,
'0' .. '9' | '+' | '-' | '.' => {
if i == 0 {
return Err(~"url: Scheme must begin with a letter.");
}
loop;
}
':' => {
if i == 0 {
return Err(~"url: Scheme cannot be empty.");
} else {
return Ok((rawurl.slice(0,i).to_owned(),
rawurl.slice(i+1,str::len(rawurl)).to_owned()));
}
}
_ => {
return Err(~"url: Invalid character in scheme.");
}
}
};
return Err(~"url: Scheme must be terminated with a colon.");
}
#[deriving(Clone, Eq)]
enum Input {
Digit, // all digits
Hex, // digits and letters a-f
Unreserved // all other legal characters
}
// returns userinfo, host, port, and unparsed part, or an error
fn get_authority(rawurl: &str) ->
Result<(Option<UserInfo>, ~str, Option<~str>, ~str), ~str> {
if !str::starts_with(rawurl, ~"//") {
// there is no authority.
return Ok((None, ~"", None, rawurl.to_str()));
}
enum State {
Start, // starting state
PassHostPort, // could be in user or port
Ip6Port, // either in ipv6 host or port
Ip6Host, // are in an ipv6 host
InHost, // are in a host - may be ipv6, but don't know yet
InPort // are in port
}
let len = rawurl.len();
let mut st = Start;
let mut in = Digit; // most restricted, start here.
let mut userinfo = None;
let mut host = ~"";
let mut port = None;
let mut colon_count = 0;
let mut pos = 0, begin = 2, end = len;
for str::each_chari(rawurl) |i,c| {
if i < 2 { loop; } // ignore the leading //
// deal with input class first
match c {
'0' .. '9' => (),
'A' .. 'F' | 'a' .. 'f' => {
if in == Digit {
in = Hex;
}
}
'G' .. 'Z' | 'g' .. 'z' | '-' | '.' | '_' | '~' | '%' |
'&' |'\'' | '(' | ')' | '+' | '!' | '*' | ',' | ';' | '=' => {
in = Unreserved;
}
':' | '@' | '?' | '#' | '/' => {
// separators, don't change anything
}
_ => {
return Err(~"Illegal character in authority");
}
}
// now process states
match c {
':' => {
colon_count += 1;
match st {
Start => {
pos = i;
st = PassHostPort;
}
PassHostPort => {
// multiple colons means ipv6 address.
if in == Unreserved {
return Err(
~"Illegal characters in IPv6 address.");
}
st = Ip6Host;
}
InHost => {
pos = i;
// can't be sure whether this is an ipv6 address or a port
if in == Unreserved {
return Err(~"Illegal characters in authority.");
}
st = Ip6Port;
}
Ip6Port => {
if in == Unreserved {
return Err(~"Illegal characters in authority.");
}
st = Ip6Host;
}
Ip6Host => {
if colon_count > 7 {
host = str::slice(rawurl, begin, i).to_owned();
pos = i;
st = InPort;
}
}
_ => {
return Err(~"Invalid ':' in authority.");
}
}
in = Digit; // reset input class
}
'@' => {
in = Digit; // reset input class
colon_count = 0; // reset count
match st {
Start => {
let user = str::slice(rawurl, begin, i).to_owned();
userinfo = Some(UserInfo::new(user, None));
st = InHost;
}
PassHostPort => {
let user = str::slice(rawurl, begin, pos).to_owned();
let pass = str::slice(rawurl, pos+1, i).to_owned();
userinfo = Some(UserInfo::new(user, Some(pass)));
st = InHost;
}
_ => {
return Err(~"Invalid '@' in authority.");
}
}
begin = i+1;
}
'?' | '#' | '/' => {
end = i;
break;
}
_ => ()
}
end = i;
}
let end = end; // make end immutable so it can be captured
let host_is_end_plus_one: &fn() -> bool = || {
end+1 == len
&& !['?', '#', '/'].contains(&(rawurl[end] as char))
};
// finish up
match st {
Start => {
if host_is_end_plus_one() {
host = str::slice(rawurl, begin, end+1).to_owned();
} else {
host = str::slice(rawurl, begin, end).to_owned();
}
}
PassHostPort | Ip6Port => {
if in != Digit {
return Err(~"Non-digit characters in port.");
}
host = str::slice(rawurl, begin, pos).to_owned();
port = Some(str::slice(rawurl, pos+1, end).to_owned());
}
Ip6Host | InHost => {
host = str::slice(rawurl, begin, end).to_owned();
}
InPort => {
if in != Digit {
return Err(~"Non-digit characters in port.");
}
port = Some(str::slice(rawurl, pos+1, end).to_owned());
}
}
let rest = if host_is_end_plus_one() { ~"" }
else { str::slice(rawurl, end, len).to_owned() };
return Ok((userinfo, host, port, rest));
}
// returns the path and unparsed part of url, or an error
fn get_path(rawurl: &str, authority: bool) ->
Result<(~str, ~str), ~str> {
let len = str::len(rawurl);
let mut end = len;
for str::each_chari(rawurl) |i,c| {
match c {
'A' .. 'Z' | 'a' .. 'z' | '0' .. '9' | '&' |'\'' | '(' | ')' | '.'
| '@' | ':' | '%' | '/' | '+' | '!' | '*' | ',' | ';' | '='
| '_' | '-' => {
loop;
}
'?' | '#' => {
end = i;
break;
}
_ => return Err(~"Invalid character in path.")
}
}
if authority {
if end != 0 && !str::starts_with(rawurl, ~"/") {
return Err(~"Non-empty path must begin with\
'/' in presence of authority.");
}
}
return Ok((decode_component(str::slice(rawurl, 0, end).to_owned()),
str::slice(rawurl, end, len).to_owned()));
}
// returns the parsed query and the fragment, if present
fn get_query_fragment(rawurl: &str) ->
Result<(Query, Option<~str>), ~str> {
if !str::starts_with(rawurl, ~"?") {
if str::starts_with(rawurl, ~"#") {
let f = decode_component(str::slice(rawurl,
1,
str::len(rawurl)).to_owned());
return Ok((~[], Some(f)));
} else {
return Ok((~[], None));
}
}
let (q, r) = split_char_first(str::slice(rawurl, 1,
str::len(rawurl)).to_owned(), '#');
let f = if str::len(r) != 0 {
Some(decode_component(r)) } else { None };
return Ok((query_from_str(q), f));
}
/**
* Parse a `str` to a `url`
*
* # Arguments
*
* `rawurl` - a string representing a full url, including scheme.
*
* # Returns
*
* a `url` that contains the parsed representation of the url.
*
*/
pub fn from_str(rawurl: &str) -> Result<Url, ~str> {
// scheme
let (scheme, rest) = match get_scheme(rawurl) {
Ok(val) => val,
Err(e) => return Err(e),
};
// authority
let (userinfo, host, port, rest) = match get_authority(rest) {
Ok(val) => val,
Err(e) => return Err(e),
};
// path
let has_authority = if host == ~"" { false } else { true };
let (path, rest) = match get_path(rest, has_authority) {
Ok(val) => val,
Err(e) => return Err(e),
};
// query and fragment
let (query, fragment) = match get_query_fragment(rest) {
Ok(val) => val,
Err(e) => return Err(e),
};
Ok(Url::new(scheme, userinfo, host, port, path, query, fragment))
}
impl FromStr for Url {
fn from_str(s: &str) -> Option<Url> {
match from_str(s) {
Ok(url) => Some(url),
Err(_) => None
}
}
}
/**
* Format a `url` as a string
*
* # Arguments
*
* `url` - a url.
*
* # Returns
*
* a `str` that contains the formatted url. Note that this will usually
* be an inverse of `from_str` but might strip out unneeded separators.
* for example, "http://somehost.com?", when parsed and formatted, will
* result in just "http://somehost.com".
*
*/
pub fn to_str(url: &Url) -> ~str {
let user = match url.user {
Some(ref user) => userinfo_to_str(user),
None => ~"",
};
let authority = if url.host.is_empty() {
~""
} else {
fmt!("//%s%s", user, url.host)
};
let query = if url.query.is_empty() {
~""
} else {
fmt!("?%s", query_to_str(&url.query))
};
let fragment = match url.fragment {
Some(ref fragment) => fmt!("#%s", encode_component(*fragment)),
None => ~"",
};
fmt!("%s:%s%s%s%s", url.scheme, authority, url.path, query, fragment)
}
impl ToStr for Url {
pub fn to_str(&self) -> ~str {
to_str(self)
}
}
#[cfg(stage0)]
impl IterBytes for Url {
fn iter_bytes(&self, lsb0: bool, f: to_bytes::Cb) {
self.to_str().iter_bytes(lsb0, f)
}
}
#[cfg(not(stage0))]
impl IterBytes for Url {
fn iter_bytes(&self, lsb0: bool, f: to_bytes::Cb) -> bool {
self.to_str().iter_bytes(lsb0, f)
}
}
// Put a few tests outside of the 'test' module so they can test the internal
// functions and those functions don't need 'pub'
#[test]
fn test_split_char_first() {
let (u,v) = split_char_first(~"hello, sweet world", ',');
assert!(u == ~"hello");
assert!(v == ~" sweet world");
let (u,v) = split_char_first(~"hello sweet world", ',');
assert!(u == ~"hello sweet world");
assert!(v == ~"");
}
#[test]
fn test_get_authority() {
let (u, h, p, r) = get_authority(
"//user:pass@rust-lang.org/something").unwrap();
assert!(u == Some(UserInfo::new(~"user", Some(~"pass"))));
assert!(h == ~"rust-lang.org");
assert!(p.is_none());
assert!(r == ~"/something");
let (u, h, p, r) = get_authority(
"//rust-lang.org:8000?something").unwrap();
assert!(u.is_none());
assert!(h == ~"rust-lang.org");
assert!(p == Some(~"8000"));
assert!(r == ~"?something");
let (u, h, p, r) = get_authority(
"//rust-lang.org#blah").unwrap();
assert!(u.is_none());
assert!(h == ~"rust-lang.org");
assert!(p.is_none());
assert!(r == ~"#blah");
// ipv6 tests
let (_, h, _, _) = get_authority(
"//2001:0db8:85a3:0042:0000:8a2e:0370:7334#blah").unwrap();
assert!(h == ~"2001:0db8:85a3:0042:0000:8a2e:0370:7334");
let (_, h, p, _) = get_authority(
"//2001:0db8:85a3:0042:0000:8a2e:0370:7334:8000#blah").unwrap();
assert!(h == ~"2001:0db8:85a3:0042:0000:8a2e:0370:7334");
assert!(p == Some(~"8000"));
let (u, h, p, _) = get_authority(
"//us:p@2001:0db8:85a3:0042:0000:8a2e:0370:7334:8000#blah"
).unwrap();
assert!(u == Some(UserInfo::new(~"us", Some(~"p"))));
assert!(h == ~"2001:0db8:85a3:0042:0000:8a2e:0370:7334");
assert!(p == Some(~"8000"));
// invalid authorities;
assert!(get_authority("//user:pass@rust-lang:something").is_err());
assert!(get_authority("//user@rust-lang:something:/path").is_err());
assert!(get_authority(
"//2001:0db8:85a3:0042:0000:8a2e:0370:7334:800a").is_err());
assert!(get_authority(
"//2001:0db8:85a3:0042:0000:8a2e:0370:7334:8000:00").is_err());
// these parse as empty, because they don't start with '//'
let (_, h, _, _) = get_authority(~"user:pass@rust-lang").unwrap();
assert!(h == ~"");
let (_, h, _, _) = get_authority(~"rust-lang.org").unwrap();
assert!(h == ~"");
}
#[test]
fn test_get_path() {
let (p, r) = get_path("/something+%20orother", true).unwrap();
assert!(p == ~"/something+ orother");
assert!(r == ~"");
let (p, r) = get_path("test@email.com#fragment", false).unwrap();
assert!(p == ~"test@email.com");
assert!(r == ~"#fragment");
let (p, r) = get_path(~"/gen/:addr=?q=v", false).unwrap();
assert!(p == ~"/gen/:addr=");
assert!(r == ~"?q=v");
//failure cases
assert!(get_path(~"something?q", true).is_err());
}
#[cfg(test)]
mod tests {
use net_url::*;
use core::hashmap::HashMap;
#[test]
fn test_url_parse() {
let url = ~"http://user:pass@rust-lang.org/doc?s=v#something";
let up = from_str(url);
let u = up.unwrap();
assert!(u.scheme == ~"http");
let userinfo = u.user.get_ref();
assert!(userinfo.user == ~"user");
assert!(userinfo.pass.get_ref() == &~"pass");
assert!(u.host == ~"rust-lang.org");
assert!(u.path == ~"/doc");
assert!(u.query == ~[(~"s", ~"v")]);
assert!(u.fragment.get_ref() == &~"something");
}
#[test]
fn test_url_parse_host_slash() {
let urlstr = ~"http://0.42.42.42/";
let url = from_str(urlstr).unwrap();
assert!(url.host == ~"0.42.42.42");
assert!(url.path == ~"/");
}
#[test]
fn test_url_with_underscores() {
let urlstr = ~"http://dotcom.com/file_name.html";
let url = from_str(urlstr).unwrap();
assert!(url.path == ~"/file_name.html");
}
#[test]
fn test_url_with_dashes() {
let urlstr = ~"http://dotcom.com/file-name.html";
let url = from_str(urlstr).unwrap();
assert!(url.path == ~"/file-name.html");
}
#[test]
fn test_no_scheme() {
assert!(get_scheme("noschemehere.html").is_err());
}
#[test]
fn test_invalid_scheme_errors() {
assert!(from_str("99://something").is_err());
assert!(from_str("://something").is_err());
}
#[test]
fn test_full_url_parse_and_format() {
let url = ~"http://user:pass@rust-lang.org/doc?s=v#something";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_userless_url_parse_and_format() {
let url = ~"http://rust-lang.org/doc?s=v#something";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_queryless_url_parse_and_format() {
let url = ~"http://user:pass@rust-lang.org/doc#something";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_empty_query_url_parse_and_format() {
let url = ~"http://user:pass@rust-lang.org/doc?#something";
let should_be = ~"http://user:pass@rust-lang.org/doc#something";
assert!(from_str(url).unwrap().to_str() == should_be);
}
#[test]
fn test_fragmentless_url_parse_and_format() {
let url = ~"http://user:pass@rust-lang.org/doc?q=v";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_minimal_url_parse_and_format() {
let url = ~"http://rust-lang.org/doc";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_scheme_host_only_url_parse_and_format() {
let url = ~"http://rust-lang.org";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_pathless_url_parse_and_format() {
let url = ~"http://user:pass@rust-lang.org?q=v#something";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_scheme_host_fragment_only_url_parse_and_format() {
let url = ~"http://rust-lang.org#something";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_url_component_encoding() {
let url = ~"http://rust-lang.org/doc%20uments?ba%25d%20=%23%26%2B";
let u = from_str(url).unwrap();
assert!(u.path == ~"/doc uments");
assert!(u.query == ~[(~"ba%d ", ~"#&+")]);
}
#[test]
fn test_url_without_authority() {
let url = ~"mailto:test@email.com";
assert!(from_str(url).unwrap().to_str() == url);
}
#[test]
fn test_encode() {
assert!(encode("") == ~"");
assert!(encode("http://example.com") == ~"http://example.com");
assert!(encode("foo bar% baz") == ~"foo%20bar%25%20baz");
assert!(encode(" ") == ~"%20");
assert!(encode("!") == ~"!");
assert!(encode("\"") == ~"\"");
assert!(encode("#") == ~"#");
assert!(encode("$") == ~"$");
assert!(encode("%") == ~"%25");
assert!(encode("&") == ~"&");
assert!(encode("'") == ~"%27");
assert!(encode("(") == ~"(");
assert!(encode(")") == ~")");
assert!(encode("*") == ~"*");
assert!(encode("+") == ~"+");
assert!(encode(",") == ~",");
assert!(encode("/") == ~"/");
assert!(encode(":") == ~":");
assert!(encode(";") == ~";");
assert!(encode("=") == ~"=");
assert!(encode("?") == ~"?");
assert!(encode("@") == ~"@");
assert!(encode("[") == ~"[");
assert!(encode("]") == ~"]");
}
#[test]
fn test_encode_component() {
assert!(encode_component("") == ~"");
assert!(encode_component("http://example.com") ==
~"http%3A%2F%2Fexample.com");
assert!(encode_component("foo bar% baz") ==
~"foo%20bar%25%20baz");
assert!(encode_component(" ") == ~"%20");
assert!(encode_component("!") == ~"%21");
assert!(encode_component("#") == ~"%23");
assert!(encode_component("$") == ~"%24");
assert!(encode_component("%") == ~"%25");
assert!(encode_component("&") == ~"%26");
assert!(encode_component("'") == ~"%27");
assert!(encode_component("(") == ~"%28");
assert!(encode_component(")") == ~"%29");
assert!(encode_component("*") == ~"%2A");
assert!(encode_component("+") == ~"%2B");
assert!(encode_component(",") == ~"%2C");
assert!(encode_component("/") == ~"%2F");
assert!(encode_component(":") == ~"%3A");
assert!(encode_component(";") == ~"%3B");
assert!(encode_component("=") == ~"%3D");
assert!(encode_component("?") == ~"%3F");
assert!(encode_component("@") == ~"%40");
assert!(encode_component("[") == ~"%5B");
assert!(encode_component("]") == ~"%5D");
}
#[test]
fn test_decode() {
assert!(decode("") == ~"");
assert!(decode("abc/def 123") == ~"abc/def 123");
assert!(decode("abc%2Fdef%20123") == ~"abc%2Fdef 123");
assert!(decode("%20") == ~" ");
assert!(decode("%21") == ~"%21");
assert!(decode("%22") == ~"%22");
assert!(decode("%23") == ~"%23");
assert!(decode("%24") == ~"%24");
assert!(decode("%25") == ~"%");
assert!(decode("%26") == ~"%26");
assert!(decode("%27") == ~"'");
assert!(decode("%28") == ~"%28");
assert!(decode("%29") == ~"%29");
assert!(decode("%2A") == ~"%2A");
assert!(decode("%2B") == ~"%2B");
assert!(decode("%2C") == ~"%2C");
assert!(decode("%2F") == ~"%2F");
assert!(decode("%3A") == ~"%3A");
assert!(decode("%3B") == ~"%3B");
assert!(decode("%3D") == ~"%3D");
assert!(decode("%3F") == ~"%3F");
assert!(decode("%40") == ~"%40");
assert!(decode("%5B") == ~"%5B");
assert!(decode("%5D") == ~"%5D");
}
#[test]
fn test_decode_component() {
assert!(decode_component("") == ~"");
assert!(decode_component("abc/def 123") == ~"abc/def 123");
assert!(decode_component("abc%2Fdef%20123") == ~"abc/def 123");
assert!(decode_component("%20") == ~" ");
assert!(decode_component("%21") == ~"!");
assert!(decode_component("%22") == ~"\"");
assert!(decode_component("%23") == ~"#");
assert!(decode_component("%24") == ~"$");
assert!(decode_component("%25") == ~"%");
assert!(decode_component("%26") == ~"&");
assert!(decode_component("%27") == ~"'");
assert!(decode_component("%28") == ~"(");
assert!(decode_component("%29") == ~")");
assert!(decode_component("%2A") == ~"*");
assert!(decode_component("%2B") == ~"+");
assert!(decode_component("%2C") == ~",");
assert!(decode_component("%2F") == ~"/");
assert!(decode_component("%3A") == ~":");
assert!(decode_component("%3B") == ~";");
assert!(decode_component("%3D") == ~"=");
assert!(decode_component("%3F") == ~"?");
assert!(decode_component("%40") == ~"@");
assert!(decode_component("%5B") == ~"[");
assert!(decode_component("%5D") == ~"]");
}
#[test]
fn test_encode_form_urlencoded() {
let mut m = HashMap::new();
assert!(encode_form_urlencoded(&m) == ~"");
m.insert(~"", ~[]);
m.insert(~"foo", ~[]);
assert!(encode_form_urlencoded(&m) == ~"");
let mut m = HashMap::new();
m.insert(~"foo", ~[~"bar", ~"123"]);
assert!(encode_form_urlencoded(&m) == ~"foo=bar&foo=123");
let mut m = HashMap::new();
m.insert(~"foo bar", ~[~"abc", ~"12 = 34"]);
assert!(encode_form_urlencoded(&m) ==
~"foo+bar=abc&foo+bar=12+%3D+34");
}
#[test]
fn test_decode_form_urlencoded() {
// FIXME #4449: Commented out because this causes an ICE, but only
// on FreeBSD
/*
assert!(decode_form_urlencoded(~[]).len() == 0);
let s = str::to_bytes("a=1&foo+bar=abc&foo+bar=12+%3D+34");
let form = decode_form_urlencoded(s);
assert!(form.len() == 2);
assert!(form.get_ref(&~"a") == &~[~"1"]);
assert!(form.get_ref(&~"foo bar") == &~[~"abc", ~"12 = 34"]);
*/
}
}