2012-12-03 18:48:01 -06:00
|
|
|
// 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.
|
|
|
|
|
2012-07-09 13:08:07 -05:00
|
|
|
//! Types/fns concerning URLs (see RFC 3986)
|
2013-01-31 19:12:29 -06:00
|
|
|
|
|
|
|
#[allow(deprecated_mode)];
|
2012-07-09 13:08:07 -05:00
|
|
|
|
2012-12-23 16:41:37 -06:00
|
|
|
use core::cmp::Eq;
|
|
|
|
use core::dvec::DVec;
|
|
|
|
use core::from_str::FromStr;
|
|
|
|
use core::io::{Reader, ReaderUtil};
|
|
|
|
use core::io;
|
2013-01-08 21:37:25 -06:00
|
|
|
use core::prelude::*;
|
2013-01-23 09:25:27 -06:00
|
|
|
use core::hashmap::linear::LinearMap;
|
2012-12-23 16:41:37 -06:00
|
|
|
use core::str;
|
|
|
|
use core::to_bytes::IterBytes;
|
|
|
|
use core::to_bytes;
|
|
|
|
use core::to_str::ToStr;
|
|
|
|
use core::to_str;
|
|
|
|
use core::uint;
|
|
|
|
use core::util;
|
|
|
|
use core::vec;
|
2012-07-24 22:21:32 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
#[deriving_eq]
|
2012-09-02 22:59:22 -05:00
|
|
|
struct Url {
|
2012-07-09 13:08:07 -05:00
|
|
|
scheme: ~str,
|
2012-08-30 13:01:39 -05:00
|
|
|
user: Option<UserInfo>,
|
2012-07-09 13:08:07 -05:00
|
|
|
host: ~str,
|
2012-08-20 14:23:37 -05:00
|
|
|
port: Option<~str>,
|
2012-07-09 13:08:07 -05:00
|
|
|
path: ~str,
|
2012-08-30 13:01:39 -05:00
|
|
|
query: Query,
|
2012-08-20 14:23:37 -05:00
|
|
|
fragment: Option<~str>
|
2012-09-02 22:59:22 -05:00
|
|
|
}
|
2012-07-09 13:08:07 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
#[deriving_eq]
|
|
|
|
struct UserInfo {
|
2012-07-09 13:08:07 -05:00
|
|
|
user: ~str,
|
2012-08-20 14:23:37 -05:00
|
|
|
pass: Option<~str>
|
2012-12-21 09:47:32 -06:00
|
|
|
}
|
2012-07-09 13:08:07 -05:00
|
|
|
|
2012-10-03 18:43:56 -05:00
|
|
|
pub type Query = ~[(~str, ~str)];
|
2012-07-09 13:08:07 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
pub impl Url {
|
|
|
|
static pure 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,
|
|
|
|
}
|
|
|
|
}
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
pub impl UserInfo {
|
|
|
|
static pure fn new(user: ~str, pass: Option<~str>) -> UserInfo {
|
|
|
|
UserInfo { user: user, pass: pass }
|
|
|
|
}
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2012-09-03 14:33:51 -05:00
|
|
|
fn encode_inner(s: &str, full_url: bool) -> ~str {
|
2012-07-24 22:21:32 -05:00
|
|
|
do io::with_str_reader(s) |rdr| {
|
|
|
|
let mut out = ~"";
|
|
|
|
|
|
|
|
while !rdr.eof() {
|
|
|
|
let ch = rdr.read_byte() as char;
|
2012-08-01 17:04:33 -05:00
|
|
|
match ch {
|
2012-07-24 22:21:32 -05:00
|
|
|
// unreserved:
|
2012-09-01 20:38:05 -05:00
|
|
|
'A' .. 'Z' |
|
|
|
|
'a' .. 'z' |
|
|
|
|
'0' .. '9' |
|
2012-08-03 21:59:04 -05:00
|
|
|
'-' | '.' | '_' | '~' => {
|
2012-09-21 20:36:32 -05:00
|
|
|
str::push_char(&mut out, ch);
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => {
|
|
|
|
if full_url {
|
2012-08-01 17:04:33 -05:00
|
|
|
match ch {
|
2012-07-24 22:21:32 -05:00
|
|
|
// gen-delims:
|
|
|
|
':' | '/' | '?' | '#' | '[' | ']' | '@' |
|
|
|
|
|
|
|
|
// sub-delims:
|
|
|
|
'!' | '$' | '&' | '"' | '(' | ')' | '*' |
|
2012-08-03 21:59:04 -05:00
|
|
|
'+' | ',' | ';' | '=' => {
|
2012-09-21 20:36:32 -05:00
|
|
|
str::push_char(&mut out, ch);
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
2012-10-12 14:32:36 -05:00
|
|
|
_ => out += fmt!("%%%X", ch as uint)
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
} else {
|
2012-10-12 14:32:36 -05:00
|
|
|
out += fmt!("%%%X", ch as uint);
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-31 22:14:50 -05:00
|
|
|
/**
|
|
|
|
* Encodes a URI by replacing reserved characters with percent encoded
|
|
|
|
* character sequences.
|
2012-07-24 22:21:32 -05:00
|
|
|
*
|
|
|
|
* This function is compliant with RFC 3986.
|
|
|
|
*/
|
2012-11-17 11:57:14 -06:00
|
|
|
pub pure fn encode(s: &str) -> ~str {
|
2012-12-21 09:47:32 -06:00
|
|
|
// FIXME(#3722): unsafe only because encode_inner does (string) IO
|
2012-11-17 11:57:14 -06:00
|
|
|
unsafe {encode_inner(s, true)}
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
2012-07-31 22:14:50 -05:00
|
|
|
/**
|
|
|
|
* Encodes a URI component by replacing reserved characters with percent
|
|
|
|
* encoded character sequences.
|
2012-07-24 22:21:32 -05:00
|
|
|
*
|
|
|
|
* This function is compliant with RFC 3986.
|
|
|
|
*/
|
2012-10-11 16:12:50 -05:00
|
|
|
|
2012-11-17 11:57:14 -06:00
|
|
|
pub pure fn encode_component(s: &str) -> ~str {
|
2012-12-21 09:47:32 -06:00
|
|
|
// FIXME(#3722): unsafe only because encode_inner does (string) IO
|
2012-11-17 11:57:14 -06:00
|
|
|
unsafe {encode_inner(s, false)}
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
2012-09-02 23:43:20 -05:00
|
|
|
fn decode_inner(s: &str, full_url: bool) -> ~str {
|
2012-07-24 22:21:32 -05:00
|
|
|
do io::with_str_reader(s) |rdr| {
|
|
|
|
let mut out = ~"";
|
|
|
|
|
|
|
|
while !rdr.eof() {
|
2012-08-01 17:04:33 -05:00
|
|
|
match rdr.read_char() {
|
2012-08-03 21:59:04 -05:00
|
|
|
'%' => {
|
2012-07-24 22:21:32 -05:00
|
|
|
let bytes = rdr.read_bytes(2u);
|
2012-09-14 11:55:33 -05:00
|
|
|
let ch = uint::parse_bytes(bytes, 16u).get() as char;
|
2012-07-24 22:21:32 -05:00
|
|
|
|
|
|
|
if full_url {
|
|
|
|
// Only decode some characters:
|
2012-08-01 17:04:33 -05:00
|
|
|
match ch {
|
2012-07-24 22:21:32 -05:00
|
|
|
// gen-delims:
|
|
|
|
':' | '/' | '?' | '#' | '[' | ']' | '@' |
|
|
|
|
|
|
|
|
// sub-delims:
|
|
|
|
'!' | '$' | '&' | '"' | '(' | ')' | '*' |
|
2012-08-03 21:59:04 -05:00
|
|
|
'+' | ',' | ';' | '=' => {
|
2012-09-21 20:36:32 -05:00
|
|
|
str::push_char(&mut out, '%');
|
|
|
|
str::push_char(&mut out, bytes[0u] as char);
|
|
|
|
str::push_char(&mut out, bytes[1u] as char);
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
2012-09-21 20:36:32 -05:00
|
|
|
ch => str::push_char(&mut out, ch)
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
} else {
|
2012-09-21 20:36:32 -05:00
|
|
|
str::push_char(&mut out, ch);
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
}
|
2012-09-21 20:36:32 -05:00
|
|
|
ch => str::push_char(&mut out, ch)
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-25 18:22:59 -05:00
|
|
|
/**
|
|
|
|
* Decode a string encoded with percent encoding.
|
2012-07-31 22:14:50 -05:00
|
|
|
*
|
2012-12-21 09:47:32 -06:00
|
|
|
* This will only decode escape sequences generated by encode.
|
2012-07-24 22:21:32 -05:00
|
|
|
*/
|
2012-11-17 11:57:14 -06:00
|
|
|
pub pure fn decode(s: &str) -> ~str {
|
2012-12-21 09:47:32 -06:00
|
|
|
// FIXME(#3722): unsafe only because decode_inner does (string) IO
|
2012-11-17 11:57:14 -06:00
|
|
|
unsafe {decode_inner(s, true)}
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
2012-07-31 22:14:50 -05:00
|
|
|
/**
|
2012-07-25 18:22:59 -05:00
|
|
|
* Decode a string encoded with percent encoding.
|
2012-07-24 22:21:32 -05:00
|
|
|
*/
|
2012-11-17 11:57:14 -06:00
|
|
|
pub pure fn decode_component(s: &str) -> ~str {
|
2012-12-21 09:47:32 -06:00
|
|
|
// FIXME(#3722): unsafe only because decode_inner does (string) IO
|
2012-11-17 11:57:14 -06:00
|
|
|
unsafe {decode_inner(s, false)}
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
2012-09-02 23:43:20 -05:00
|
|
|
fn encode_plus(s: &str) -> ~str {
|
2012-07-24 22:21:32 -05:00
|
|
|
do io::with_str_reader(s) |rdr| {
|
|
|
|
let mut out = ~"";
|
|
|
|
|
|
|
|
while !rdr.eof() {
|
|
|
|
let ch = rdr.read_byte() as char;
|
2012-08-01 17:04:33 -05:00
|
|
|
match ch {
|
2012-09-01 20:38:05 -05:00
|
|
|
'A' .. 'Z' | 'a' .. 'z' | '0' .. '9' | '_' | '.' | '-' => {
|
2012-09-21 20:36:32 -05:00
|
|
|
str::push_char(&mut out, ch);
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
2012-09-21 20:36:32 -05:00
|
|
|
' ' => str::push_char(&mut out, '+'),
|
2012-10-12 14:32:36 -05:00
|
|
|
_ => out += fmt!("%%%X", ch as uint)
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-31 22:14:50 -05:00
|
|
|
/**
|
2012-07-25 18:22:59 -05:00
|
|
|
* Encode a hashmap to the 'application/x-www-form-urlencoded' media type.
|
2012-07-24 22:21:32 -05:00
|
|
|
*/
|
2012-12-21 09:47:32 -06:00
|
|
|
pub fn encode_form_urlencoded(m: &LinearMap<~str, ~[~str]>) -> ~str {
|
2012-07-24 22:21:32 -05:00
|
|
|
let mut out = ~"";
|
|
|
|
let mut first = true;
|
|
|
|
|
2013-02-07 20:03:13 -06:00
|
|
|
for m.each |&(key, values)| {
|
2012-12-21 09:47:32 -06:00
|
|
|
let key = encode_plus(*key);
|
2012-07-24 22:21:32 -05:00
|
|
|
|
2013-02-07 20:03:13 -06:00
|
|
|
for values.each |value| {
|
2012-07-24 22:21:32 -05:00
|
|
|
if first {
|
|
|
|
first = false;
|
|
|
|
} else {
|
2012-09-21 20:36:32 -05:00
|
|
|
str::push_char(&mut out, '&');
|
2012-07-24 22:21:32 -05:00
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
out += fmt!("%s=%s", key, encode_plus(*value));
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out
|
|
|
|
}
|
|
|
|
|
2012-07-31 22:14:50 -05:00
|
|
|
/**
|
2012-07-25 18:22:59 -05:00
|
|
|
* Decode a string encoded with the 'application/x-www-form-urlencoded' media
|
2012-07-24 22:21:32 -05:00
|
|
|
* type into a hashmap.
|
|
|
|
*/
|
2013-01-23 10:08:33 -06:00
|
|
|
pub fn decode_form_urlencoded(s: &[u8]) -> LinearMap<~str, ~[~str]> {
|
2012-07-24 22:21:32 -05:00
|
|
|
do io::with_bytes_reader(s) |rdr| {
|
2013-01-23 16:06:32 -06:00
|
|
|
let mut m = LinearMap::new();
|
2012-07-24 22:21:32 -05:00
|
|
|
let mut key = ~"";
|
|
|
|
let mut value = ~"";
|
|
|
|
let mut parsing_key = true;
|
|
|
|
|
|
|
|
while !rdr.eof() {
|
2012-08-01 17:04:33 -05:00
|
|
|
match rdr.read_char() {
|
2012-12-21 09:47:32 -06:00
|
|
|
'&' | ';' => {
|
|
|
|
if key != ~"" && value != ~"" {
|
|
|
|
let mut values = match m.pop(&key) {
|
2013-02-15 01:30:30 -06:00
|
|
|
Some(values) => values,
|
2012-12-21 09:47:32 -06:00
|
|
|
None => ~[],
|
|
|
|
};
|
|
|
|
|
|
|
|
values.push(value);
|
2012-07-24 22:21:32 -05:00
|
|
|
m.insert(key, values);
|
2012-12-21 09:47:32 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
parsing_key = true;
|
|
|
|
key = ~"";
|
|
|
|
value = ~"";
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
2012-12-21 09:47:32 -06:00
|
|
|
'=' => parsing_key = false,
|
|
|
|
ch => {
|
|
|
|
let ch = match ch {
|
|
|
|
'%' => {
|
|
|
|
let bytes = rdr.read_bytes(2u);
|
|
|
|
uint::parse_bytes(bytes, 16u).get() as char
|
|
|
|
}
|
|
|
|
'+' => ' ',
|
|
|
|
ch => ch
|
|
|
|
};
|
2012-07-24 22:21:32 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
if parsing_key {
|
|
|
|
str::push_char(&mut key, ch)
|
|
|
|
} else {
|
|
|
|
str::push_char(&mut value, ch)
|
|
|
|
}
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if key != ~"" && value != ~"" {
|
2012-12-21 09:47:32 -06:00
|
|
|
let mut values = match m.pop(&key) {
|
2013-02-15 01:30:30 -06:00
|
|
|
Some(values) => values,
|
2012-12-21 09:47:32 -06:00
|
|
|
None => ~[],
|
2012-07-24 22:21:32 -05:00
|
|
|
};
|
2012-12-21 09:47:32 -06:00
|
|
|
|
|
|
|
values.push(value);
|
|
|
|
m.insert(key, values);
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
m
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-17 11:57:14 -06:00
|
|
|
pure fn split_char_first(s: &str, c: char) -> (~str, ~str) {
|
2012-07-25 21:40:31 -05:00
|
|
|
let len = str::len(s);
|
|
|
|
let mut index = len;
|
2012-08-01 17:04:33 -05:00
|
|
|
let mut mat = 0;
|
2012-12-21 09:47:32 -06:00
|
|
|
// FIXME(#3722): unsafe only because decode_inner does (string) IO
|
2012-11-17 11:57:14 -06:00
|
|
|
unsafe {
|
|
|
|
do io::with_str_reader(s) |rdr| {
|
2012-12-21 09:47:32 -06:00
|
|
|
let mut ch;
|
2012-11-17 11:57:14 -06:00
|
|
|
while !rdr.eof() {
|
|
|
|
ch = rdr.read_byte() as char;
|
|
|
|
if ch == c {
|
|
|
|
// found a match, adjust markers
|
|
|
|
index = rdr.tell()-1;
|
|
|
|
mat = 1;
|
|
|
|
break;
|
|
|
|
}
|
2012-07-25 21:40:31 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-08-01 17:04:33 -05:00
|
|
|
if index+mat == len {
|
2012-07-25 21:40:31 -05:00
|
|
|
return (str::slice(s, 0, index), ~"");
|
2012-07-09 13:08:07 -05:00
|
|
|
} else {
|
2012-07-31 22:14:50 -05:00
|
|
|
return (str::slice(s, 0, index),
|
2012-08-01 17:04:33 -05:00
|
|
|
str::slice(s, index + mat, str::len(s)));
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-17 11:57:14 -06:00
|
|
|
pure fn userinfo_from_str(uinfo: &str) -> UserInfo {
|
2012-07-09 13:08:07 -05:00
|
|
|
let (user, p) = split_char_first(uinfo, ':');
|
|
|
|
let pass = if str::len(p) == 0 {
|
2012-12-21 09:47:32 -06:00
|
|
|
None
|
2012-07-09 13:08:07 -05:00
|
|
|
} else {
|
2012-12-21 09:47:32 -06:00
|
|
|
Some(p)
|
2012-07-09 13:08:07 -05:00
|
|
|
};
|
2012-12-21 09:47:32 -06:00
|
|
|
return UserInfo::new(user, pass);
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
pure fn userinfo_to_str(userinfo: &UserInfo) -> ~str {
|
|
|
|
match userinfo.pass {
|
|
|
|
Some(ref pass) => fmt!("%s:%s@", userinfo.user, *pass),
|
|
|
|
None => fmt!("%s@", userinfo.user),
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-17 11:57:14 -06:00
|
|
|
pure fn query_from_str(rawquery: &str) -> Query {
|
2012-08-30 13:01:39 -05:00
|
|
|
let mut query: Query = ~[];
|
2012-07-09 13:08:07 -05:00
|
|
|
if str::len(rawquery) != 0 {
|
|
|
|
for str::split_char(rawquery, '&').each |p| {
|
2012-09-19 18:55:01 -05:00
|
|
|
let (k, v) = split_char_first(*p, '=');
|
2012-12-21 09:47:32 -06:00
|
|
|
// FIXME(#3722): unsafe only because decode_inner does (string) IO
|
2012-11-17 11:57:14 -06:00
|
|
|
unsafe {query.push((decode_component(k), decode_component(v)));}
|
2012-07-09 13:08:07 -05:00
|
|
|
};
|
|
|
|
}
|
2012-08-01 19:30:05 -05:00
|
|
|
return query;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2013-01-23 13:43:58 -06:00
|
|
|
pub pure fn query_to_str(query: &Query) -> ~str {
|
|
|
|
unsafe {
|
|
|
|
// FIXME(#3722): unsafe only because decode_inner does (string) IO
|
|
|
|
let mut strvec = ~[];
|
|
|
|
for query.each |kv| {
|
|
|
|
match kv {
|
|
|
|
&(ref k, ref v) => {
|
|
|
|
strvec.push(fmt!("%s=%s",
|
|
|
|
encode_component(*k),
|
|
|
|
encode_component(*v))
|
|
|
|
);
|
|
|
|
}
|
2012-12-21 09:47:32 -06:00
|
|
|
}
|
2012-10-11 16:12:50 -05:00
|
|
|
}
|
2013-01-23 13:43:58 -06:00
|
|
|
return str::connect(strvec, ~"&");
|
2012-12-21 09:47:32 -06:00
|
|
|
}
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2012-07-31 21:44:37 -05:00
|
|
|
// returns the scheme and the rest of the url, or a parsing error
|
2012-12-21 09:47:32 -06:00
|
|
|
pub pure fn get_scheme(rawurl: &str) -> Result<(~str, ~str), ~str> {
|
2012-07-09 13:08:07 -05:00
|
|
|
for str::each_chari(rawurl) |i,c| {
|
2012-08-01 17:04:33 -05:00
|
|
|
match c {
|
2012-09-07 17:32:04 -05:00
|
|
|
'A' .. 'Z' | 'a' .. 'z' => loop,
|
2012-09-01 20:38:05 -05:00
|
|
|
'0' .. '9' | '+' | '-' | '.' => {
|
2012-07-25 21:40:31 -05:00
|
|
|
if i == 0 {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"url: Scheme must begin with a letter.");
|
2012-07-25 21:40:31 -05:00
|
|
|
}
|
2012-09-07 17:32:04 -05:00
|
|
|
loop;
|
2012-07-25 21:40:31 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
':' => {
|
2012-07-25 21:40:31 -05:00
|
|
|
if i == 0 {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"url: Scheme cannot be empty.");
|
2012-07-25 21:40:31 -05:00
|
|
|
} else {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Ok((rawurl.slice(0,i),
|
2012-07-31 21:44:37 -05:00
|
|
|
rawurl.slice(i+1,str::len(rawurl))));
|
2012-07-25 21:40:31 -05:00
|
|
|
}
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"url: Invalid character in scheme.");
|
2012-07-25 21:40:31 -05:00
|
|
|
}
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
};
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"url: Scheme must be terminated with a colon.");
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
#[deriving_eq]
|
2012-08-30 13:01:39 -05:00
|
|
|
enum Input {
|
|
|
|
Digit, // all digits
|
|
|
|
Hex, // digits and letters a-f
|
|
|
|
Unreserved // all other legal characters
|
2012-08-27 18:26:35 -05:00
|
|
|
}
|
|
|
|
|
2012-07-31 21:44:37 -05:00
|
|
|
// returns userinfo, host, port, and unparsed part, or an error
|
2012-11-17 11:57:14 -06:00
|
|
|
pure fn get_authority(rawurl: &str) ->
|
2012-12-21 09:47:32 -06:00
|
|
|
Result<(Option<UserInfo>, ~str, Option<~str>, ~str), ~str> {
|
2012-07-31 21:44:37 -05:00
|
|
|
if !str::starts_with(rawurl, ~"//") {
|
|
|
|
// there is no authority.
|
2012-12-21 09:47:32 -06:00
|
|
|
return Ok((None, ~"", None, rawurl.to_str()));
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
|
2012-08-30 13:01:39 -05:00
|
|
|
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
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-27 18:26:35 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
let len = rawurl.len();
|
|
|
|
let mut st = Start;
|
|
|
|
let mut in = Digit; // most restricted, start here.
|
2012-07-31 21:44:37 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
let mut userinfo = None;
|
|
|
|
let mut host = ~"";
|
|
|
|
let mut port = None;
|
2012-07-31 21:44:37 -05:00
|
|
|
|
|
|
|
let mut colon_count = 0;
|
2012-12-21 09:47:32 -06:00
|
|
|
let mut pos = 0, begin = 2, end = len;
|
2012-07-31 21:44:37 -05:00
|
|
|
|
2012-08-01 16:19:43 -05:00
|
|
|
for str::each_chari(rawurl) |i,c| {
|
2012-09-07 17:32:04 -05:00
|
|
|
if i < 2 { loop; } // ignore the leading //
|
2012-07-31 21:44:37 -05:00
|
|
|
|
|
|
|
// deal with input class first
|
2012-08-01 17:04:33 -05:00
|
|
|
match c {
|
2012-09-01 20:38:05 -05:00
|
|
|
'0' .. '9' => (),
|
|
|
|
'A' .. 'F' | 'a' .. 'f' => {
|
2012-08-30 13:01:39 -05:00
|
|
|
if in == Digit {
|
|
|
|
in = Hex;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
2012-09-01 20:38:05 -05:00
|
|
|
'G' .. 'Z' | 'g' .. 'z' | '-' | '.' | '_' | '~' | '%' |
|
2012-08-03 21:59:04 -05:00
|
|
|
'&' |'\'' | '(' | ')' | '+' | '!' | '*' | ',' | ';' | '=' => {
|
2012-08-30 13:01:39 -05:00
|
|
|
in = Unreserved;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
':' | '@' | '?' | '#' | '/' => {
|
2012-07-31 21:44:37 -05:00
|
|
|
// separators, don't change anything
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"Illegal character in authority");
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-31 22:14:50 -05:00
|
|
|
// now process states
|
2012-08-01 17:04:33 -05:00
|
|
|
match c {
|
2012-08-03 21:59:04 -05:00
|
|
|
':' => {
|
2012-07-31 21:44:37 -05:00
|
|
|
colon_count += 1;
|
2012-08-01 17:04:33 -05:00
|
|
|
match st {
|
2012-08-30 13:01:39 -05:00
|
|
|
Start => {
|
2012-07-31 21:44:37 -05:00
|
|
|
pos = i;
|
2012-08-30 13:01:39 -05:00
|
|
|
st = PassHostPort;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
PassHostPort => {
|
2012-07-31 21:44:37 -05:00
|
|
|
// multiple colons means ipv6 address.
|
2012-08-30 13:01:39 -05:00
|
|
|
if in == Unreserved {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(
|
|
|
|
~"Illegal characters in IPv6 address.");
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
st = Ip6Host;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
InHost => {
|
2012-07-31 21:44:37 -05:00
|
|
|
pos = i;
|
|
|
|
// can't be sure whether this is an ipv6 address or a port
|
2012-08-30 13:01:39 -05:00
|
|
|
if in == Unreserved {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"Illegal characters in authority.");
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
st = Ip6Port;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
Ip6Port => {
|
|
|
|
if in == Unreserved {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"Illegal characters in authority.");
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
st = Ip6Host;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
Ip6Host => {
|
2012-07-31 21:44:37 -05:00
|
|
|
if colon_count > 7 {
|
|
|
|
host = str::slice(rawurl, begin, i);
|
|
|
|
pos = i;
|
2012-08-30 13:01:39 -05:00
|
|
|
st = InPort;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"Invalid ':' in authority.");
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
in = Digit; // reset input class
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
|
2012-08-03 21:59:04 -05:00
|
|
|
'@' => {
|
2012-08-30 13:01:39 -05:00
|
|
|
in = Digit; // reset input class
|
2012-07-31 21:44:37 -05:00
|
|
|
colon_count = 0; // reset count
|
2012-08-01 17:04:33 -05:00
|
|
|
match st {
|
2012-08-30 13:01:39 -05:00
|
|
|
Start => {
|
2012-07-31 21:44:37 -05:00
|
|
|
let user = str::slice(rawurl, begin, i);
|
2012-12-21 09:47:32 -06:00
|
|
|
userinfo = Some(UserInfo::new(user, None));
|
2012-08-30 13:01:39 -05:00
|
|
|
st = InHost;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
PassHostPort => {
|
2012-07-31 21:44:37 -05:00
|
|
|
let user = str::slice(rawurl, begin, pos);
|
|
|
|
let pass = str::slice(rawurl, pos+1, i);
|
2012-12-21 09:47:32 -06:00
|
|
|
userinfo = Some(UserInfo::new(user, Some(pass)));
|
2012-08-30 13:01:39 -05:00
|
|
|
st = InHost;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"Invalid '@' in authority.");
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
begin = i+1;
|
|
|
|
}
|
2012-07-31 22:14:50 -05:00
|
|
|
|
2012-08-03 21:59:04 -05:00
|
|
|
'?' | '#' | '/' => {
|
2012-08-01 16:19:43 -05:00
|
|
|
end = i;
|
2012-07-31 21:44:37 -05:00
|
|
|
break;
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => ()
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-01 16:19:43 -05:00
|
|
|
end = i;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
|
2012-08-05 18:33:28 -05:00
|
|
|
let end = end; // make end immutable so it can be captured
|
|
|
|
|
2012-11-17 11:57:14 -06:00
|
|
|
let host_is_end_plus_one: &pure fn() -> bool = || {
|
2012-08-05 18:33:28 -05:00
|
|
|
end+1 == len
|
2012-09-25 19:39:22 -05:00
|
|
|
&& !['?', '#', '/'].contains(&(rawurl[end] as char))
|
2012-08-05 18:33:28 -05:00
|
|
|
};
|
|
|
|
|
2012-07-31 21:44:37 -05:00
|
|
|
// finish up
|
2012-08-01 17:04:33 -05:00
|
|
|
match st {
|
2012-08-30 13:01:39 -05:00
|
|
|
Start => {
|
2012-08-05 18:33:28 -05:00
|
|
|
if host_is_end_plus_one() {
|
2012-08-01 16:19:43 -05:00
|
|
|
host = str::slice(rawurl, begin, end+1);
|
2012-07-31 21:44:37 -05:00
|
|
|
} else {
|
2012-08-01 16:19:43 -05:00
|
|
|
host = str::slice(rawurl, begin, end);
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
PassHostPort | Ip6Port => {
|
|
|
|
if in != Digit {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"Non-digit characters in port.");
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
host = str::slice(rawurl, begin, pos);
|
2012-12-21 09:47:32 -06:00
|
|
|
port = Some(str::slice(rawurl, pos+1, end));
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
Ip6Host | InHost => {
|
2012-08-01 16:19:43 -05:00
|
|
|
host = str::slice(rawurl, begin, end);
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-30 13:01:39 -05:00
|
|
|
InPort => {
|
|
|
|
if in != Digit {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"Non-digit characters in port.");
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-12-21 09:47:32 -06:00
|
|
|
port = Some(str::slice(rawurl, pos+1, end));
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-05 18:33:28 -05:00
|
|
|
let rest = if host_is_end_plus_one() { ~"" }
|
2012-08-01 16:19:43 -05:00
|
|
|
else { str::slice(rawurl, end, len) };
|
2012-12-21 09:47:32 -06:00
|
|
|
return Ok((userinfo, host, port, rest));
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// returns the path and unparsed part of url, or an error
|
2012-12-21 09:47:32 -06:00
|
|
|
pure fn get_path(rawurl: &str, authority: bool) ->
|
|
|
|
Result<(~str, ~str), ~str> {
|
2012-07-31 21:44:37 -05:00
|
|
|
let len = str::len(rawurl);
|
|
|
|
let mut end = len;
|
|
|
|
for str::each_chari(rawurl) |i,c| {
|
2012-08-01 17:04:33 -05:00
|
|
|
match c {
|
2012-09-01 20:38:05 -05:00
|
|
|
'A' .. 'Z' | 'a' .. 'z' | '0' .. '9' | '&' |'\'' | '(' | ')' | '.'
|
2012-08-06 15:12:49 -05:00
|
|
|
| '@' | ':' | '%' | '/' | '+' | '!' | '*' | ',' | ';' | '='
|
2012-08-06 17:17:08 -05:00
|
|
|
| '_' | '-' => {
|
2012-09-07 17:32:04 -05:00
|
|
|
loop;
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
'?' | '#' => {
|
2012-07-31 21:44:37 -05:00
|
|
|
end = i;
|
|
|
|
break;
|
|
|
|
}
|
2012-12-21 09:47:32 -06:00
|
|
|
_ => return Err(~"Invalid character in path.")
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if authority {
|
|
|
|
if end != 0 && !str::starts_with(rawurl, ~"/") {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Err(~"Non-empty path must begin with\
|
2012-07-31 21:44:37 -05:00
|
|
|
'/' in presence of authority.");
|
|
|
|
}
|
|
|
|
}
|
2012-07-31 22:14:50 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
return Ok((decode_component(str::slice(rawurl, 0, end)),
|
2012-07-31 21:44:37 -05:00
|
|
|
str::slice(rawurl, end, len)));
|
|
|
|
}
|
|
|
|
|
|
|
|
// returns the parsed query and the fragment, if present
|
2012-11-17 11:57:14 -06:00
|
|
|
pure fn get_query_fragment(rawurl: &str) ->
|
2012-12-21 09:47:32 -06:00
|
|
|
Result<(Query, Option<~str>), ~str> {
|
2012-07-31 21:44:37 -05:00
|
|
|
if !str::starts_with(rawurl, ~"?") {
|
|
|
|
if str::starts_with(rawurl, ~"#") {
|
2012-07-31 22:14:50 -05:00
|
|
|
let f = decode_component(str::slice(rawurl,
|
|
|
|
1,
|
2012-07-31 21:44:37 -05:00
|
|
|
str::len(rawurl)));
|
2012-12-21 09:47:32 -06:00
|
|
|
return Ok((~[], Some(f)));
|
2012-07-31 21:44:37 -05:00
|
|
|
} else {
|
2012-12-21 09:47:32 -06:00
|
|
|
return Ok((~[], None));
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
}
|
2012-07-31 22:14:50 -05:00
|
|
|
let (q, r) = split_char_first(str::slice(rawurl, 1,
|
2012-07-31 21:44:37 -05:00
|
|
|
str::len(rawurl)), '#');
|
2012-07-31 22:14:50 -05:00
|
|
|
let f = if str::len(r) != 0 {
|
2012-12-21 09:47:32 -06:00
|
|
|
Some(decode_component(r)) } else { None };
|
|
|
|
return Ok((query_from_str(q), f));
|
2012-07-31 21:44:37 -05:00
|
|
|
}
|
|
|
|
|
2012-07-09 13:08:07 -05:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
pub pure fn from_str(rawurl: &str) -> Result<Url, ~str> {
|
2012-07-31 21:44:37 -05:00
|
|
|
// scheme
|
2012-12-21 09:47:32 -06:00
|
|
|
let (scheme, rest) = match get_scheme(rawurl) {
|
|
|
|
Ok(val) => val,
|
|
|
|
Err(e) => return Err(e),
|
|
|
|
};
|
2012-07-31 21:44:37 -05:00
|
|
|
|
|
|
|
// authority
|
2012-12-21 09:47:32 -06:00
|
|
|
let (userinfo, host, port, rest) = match get_authority(rest) {
|
|
|
|
Ok(val) => val,
|
|
|
|
Err(e) => return Err(e),
|
|
|
|
};
|
2012-07-31 21:44:37 -05:00
|
|
|
|
|
|
|
// path
|
|
|
|
let has_authority = if host == ~"" { false } else { true };
|
2012-12-21 09:47:32 -06:00
|
|
|
let (path, rest) = match get_path(rest, has_authority) {
|
|
|
|
Ok(val) => val,
|
|
|
|
Err(e) => return Err(e),
|
|
|
|
};
|
2012-07-31 21:44:37 -05:00
|
|
|
|
|
|
|
// query and fragment
|
2012-12-21 09:47:32 -06:00
|
|
|
let (query, fragment) = match get_query_fragment(rest) {
|
|
|
|
Ok(val) => val,
|
|
|
|
Err(e) => return Err(e),
|
|
|
|
};
|
2012-07-09 13:08:07 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
Ok(Url::new(scheme, userinfo, host, port, path, query, fragment))
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2013-02-14 13:47:00 -06:00
|
|
|
impl FromStr for Url {
|
2012-11-17 11:57:14 -06:00
|
|
|
static pure fn from_str(s: &str) -> Option<Url> {
|
2012-09-03 19:01:34 -05:00
|
|
|
match from_str(s) {
|
2013-02-15 01:30:30 -06:00
|
|
|
Ok(url) => Some(url),
|
2012-09-03 19:01:34 -05:00
|
|
|
Err(_) => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-09 13:08:07 -05:00
|
|
|
/**
|
|
|
|
* 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".
|
|
|
|
*
|
|
|
|
*/
|
2012-12-21 09:47:32 -06:00
|
|
|
pub pure fn to_str(url: &Url) -> ~str {
|
|
|
|
let user = match url.user {
|
|
|
|
Some(ref user) => userinfo_to_str(user),
|
|
|
|
None => ~"",
|
2012-07-09 13:08:07 -05:00
|
|
|
};
|
2012-12-21 09:47:32 -06:00
|
|
|
|
|
|
|
let authority = if url.host.is_empty() {
|
2012-07-31 23:40:38 -05:00
|
|
|
~""
|
2012-12-21 09:47:32 -06:00
|
|
|
} else {
|
|
|
|
fmt!("//%s%s", user, url.host)
|
2012-07-31 23:40:38 -05:00
|
|
|
};
|
2012-12-21 09:47:32 -06:00
|
|
|
|
|
|
|
let query = if url.query.is_empty() {
|
2012-07-09 13:08:07 -05:00
|
|
|
~""
|
|
|
|
} else {
|
2012-12-21 09:47:32 -06:00
|
|
|
fmt!("?%s", query_to_str(&url.query))
|
2012-07-09 13:08:07 -05:00
|
|
|
};
|
2012-12-21 09:47:32 -06:00
|
|
|
|
|
|
|
let fragment = match url.fragment {
|
|
|
|
Some(ref fragment) => fmt!("#%s", encode_component(*fragment)),
|
|
|
|
None => ~"",
|
2012-07-09 13:08:07 -05:00
|
|
|
};
|
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
fmt!("%s:%s%s%s%s", url.scheme, authority, url.path, query, fragment)
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2013-02-14 13:47:00 -06:00
|
|
|
impl to_str::ToStr for Url {
|
2013-02-03 22:47:26 -06:00
|
|
|
pub pure fn to_str(&self) -> ~str {
|
|
|
|
to_str(self)
|
2012-07-28 18:05:06 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-14 13:47:00 -06:00
|
|
|
impl to_bytes::IterBytes for Url {
|
2012-11-28 13:36:04 -06:00
|
|
|
pure fn iter_bytes(&self, lsb0: bool, f: to_bytes::Cb) {
|
2012-12-21 09:47:32 -06:00
|
|
|
self.to_str().iter_bytes(lsb0, f)
|
2012-11-28 13:36:04 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-09 13:08:07 -05:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2013-01-08 21:37:25 -06:00
|
|
|
use core::prelude::*;
|
|
|
|
|
|
|
|
use net_url::*;
|
2012-12-28 14:46:08 -06:00
|
|
|
use net_url::UserInfo;
|
|
|
|
|
|
|
|
use core::result;
|
|
|
|
use core::str;
|
|
|
|
|
2012-07-25 21:40:31 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_split_char_first() {
|
2012-07-25 21:40:31 -05:00
|
|
|
let (u,v) = split_char_first(~"hello, sweet world", ',');
|
|
|
|
assert u == ~"hello";
|
|
|
|
assert v == ~" sweet world";
|
2012-07-31 22:14:50 -05:00
|
|
|
|
2012-07-31 21:44:37 -05:00
|
|
|
let (u,v) = split_char_first(~"hello sweet world", ',');
|
|
|
|
assert u == ~"hello sweet world";
|
|
|
|
assert v == ~"";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_get_authority() {
|
2012-12-21 09:47:32 -06:00
|
|
|
let (u, h, p, r) = get_authority(
|
|
|
|
"//user:pass@rust-lang.org/something").unwrap();
|
|
|
|
assert u == Some(UserInfo::new(~"user", Some(~"pass")));
|
2012-07-31 21:44:37 -05:00
|
|
|
assert h == ~"rust-lang.org";
|
2012-09-21 21:37:57 -05:00
|
|
|
assert p.is_none();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert r == ~"/something";
|
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
let (u, h, p, r) = get_authority(
|
|
|
|
"//rust-lang.org:8000?something").unwrap();
|
2012-09-21 21:37:57 -05:00
|
|
|
assert u.is_none();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert h == ~"rust-lang.org";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert p == Some(~"8000");
|
2012-07-31 21:44:37 -05:00
|
|
|
assert r == ~"?something";
|
2012-07-31 22:14:50 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
let (u, h, p, r) = get_authority(
|
|
|
|
"//rust-lang.org#blah").unwrap();
|
2012-09-21 21:37:57 -05:00
|
|
|
assert u.is_none();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert h == ~"rust-lang.org";
|
2012-09-21 21:37:57 -05:00
|
|
|
assert p.is_none();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert r == ~"#blah";
|
|
|
|
|
|
|
|
// ipv6 tests
|
2012-12-21 09:47:32 -06:00
|
|
|
let (_, h, _, _) = get_authority(
|
|
|
|
"//2001:0db8:85a3:0042:0000:8a2e:0370:7334#blah").unwrap();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert h == ~"2001:0db8:85a3:0042:0000:8a2e:0370:7334";
|
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
let (_, h, p, _) = get_authority(
|
|
|
|
"//2001:0db8:85a3:0042:0000:8a2e:0370:7334:8000#blah").unwrap();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert h == ~"2001:0db8:85a3:0042:0000:8a2e:0370:7334";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert p == Some(~"8000");
|
2012-07-31 21:44:37 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
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")));
|
2012-07-31 21:44:37 -05:00
|
|
|
assert h == ~"2001:0db8:85a3:0042:0000:8a2e:0370:7334";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert p == Some(~"8000");
|
2012-07-31 21:44:37 -05:00
|
|
|
|
2012-07-31 22:14:50 -05:00
|
|
|
// invalid authorities;
|
2012-12-21 09:47:32 -06:00
|
|
|
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();
|
2012-07-31 21:44:37 -05:00
|
|
|
|
|
|
|
// these parse as empty, because they don't start with '//'
|
2012-12-21 09:47:32 -06:00
|
|
|
let (_, h, _, _) = get_authority(~"user:pass@rust-lang").unwrap();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert h == ~"";
|
2012-12-21 09:47:32 -06:00
|
|
|
let (_, h, _, _) = get_authority(~"rust-lang.org").unwrap();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert h == ~"";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_get_path() {
|
2012-12-21 09:47:32 -06:00
|
|
|
let (p, r) = get_path("/something+%20orother", true).unwrap();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert p == ~"/something+ orother";
|
|
|
|
assert r == ~"";
|
2012-12-21 09:47:32 -06:00
|
|
|
let (p, r) = get_path("test@email.com#fragment", false).unwrap();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert p == ~"test@email.com";
|
|
|
|
assert r == ~"#fragment";
|
2012-12-21 09:47:32 -06:00
|
|
|
let (p, r) = get_path(~"/gen/:addr=?q=v", false).unwrap();
|
2012-07-31 21:44:37 -05:00
|
|
|
assert p == ~"/gen/:addr=";
|
|
|
|
assert r == ~"?q=v";
|
2012-07-31 22:14:50 -05:00
|
|
|
|
2012-07-31 21:44:37 -05:00
|
|
|
//failure cases
|
2012-12-21 09:47:32 -06:00
|
|
|
assert get_path(~"something?q", true).is_err();
|
2012-07-25 21:40:31 -05:00
|
|
|
}
|
|
|
|
|
2012-07-25 18:22:59 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_url_parse() {
|
2012-07-25 18:22:59 -05:00
|
|
|
let url = ~"http://user:pass@rust-lang.org/doc?s=v#something";
|
2012-07-31 22:14:50 -05:00
|
|
|
|
2012-07-31 21:44:37 -05:00
|
|
|
let up = from_str(url);
|
2012-12-21 09:47:32 -06:00
|
|
|
let u = up.unwrap();
|
2012-07-25 18:22:59 -05:00
|
|
|
assert u.scheme == ~"http";
|
2012-12-21 09:47:32 -06:00
|
|
|
let userinfo = u.user.get_ref();
|
|
|
|
assert userinfo.user == ~"user";
|
|
|
|
assert userinfo.pass.get_ref() == &~"pass";
|
2012-07-25 18:22:59 -05:00
|
|
|
assert u.host == ~"rust-lang.org";
|
|
|
|
assert u.path == ~"/doc";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert u.query == ~[(~"s", ~"v")];
|
|
|
|
assert u.fragment.get_ref() == &~"something";
|
2012-07-25 18:22:59 -05:00
|
|
|
}
|
2012-07-25 21:40:31 -05:00
|
|
|
|
2012-08-05 18:33:28 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_url_parse_host_slash() {
|
2012-08-05 18:33:28 -05:00
|
|
|
let urlstr = ~"http://0.42.42.42/";
|
2012-12-21 09:47:32 -06:00
|
|
|
let url = from_str(urlstr).unwrap();
|
2012-08-05 18:33:28 -05:00
|
|
|
assert url.host == ~"0.42.42.42";
|
|
|
|
assert url.path == ~"/";
|
|
|
|
}
|
|
|
|
|
2012-08-06 15:12:49 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_url_with_underscores() {
|
2012-08-06 15:12:49 -05:00
|
|
|
let urlstr = ~"http://dotcom.com/file_name.html";
|
2012-12-21 09:47:32 -06:00
|
|
|
let url = from_str(urlstr).unwrap();
|
2012-08-06 15:12:49 -05:00
|
|
|
assert url.path == ~"/file_name.html";
|
|
|
|
}
|
|
|
|
|
2012-08-06 17:17:08 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_url_with_dashes() {
|
2012-08-06 17:17:08 -05:00
|
|
|
let urlstr = ~"http://dotcom.com/file-name.html";
|
2012-12-21 09:47:32 -06:00
|
|
|
let url = from_str(urlstr).unwrap();
|
2012-08-06 17:17:08 -05:00
|
|
|
assert url.path == ~"/file-name.html";
|
|
|
|
}
|
|
|
|
|
2012-08-05 15:48:26 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_no_scheme() {
|
2012-12-21 09:47:32 -06:00
|
|
|
assert get_scheme("noschemehere.html").is_err();
|
2012-08-05 15:48:26 -05:00
|
|
|
}
|
|
|
|
|
2012-07-25 21:40:31 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_invalid_scheme_errors() {
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str("99://something").is_err();
|
|
|
|
assert from_str("://something").is_err();
|
2012-07-25 21:40:31 -05:00
|
|
|
}
|
|
|
|
|
2012-07-09 13:08:07 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_full_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://user:pass@rust-lang.org/doc?s=v#something";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_userless_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://rust-lang.org/doc?s=v#something";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_queryless_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://user:pass@rust-lang.org/doc#something";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_empty_query_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://user:pass@rust-lang.org/doc?#something";
|
|
|
|
let should_be = ~"http://user:pass@rust-lang.org/doc#something";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == should_be;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_fragmentless_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://user:pass@rust-lang.org/doc?q=v";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_minimal_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://rust-lang.org/doc";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_scheme_host_only_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://rust-lang.org";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_pathless_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://user:pass@rust-lang.org?q=v#something";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_scheme_host_fragment_only_url_parse_and_format() {
|
2012-07-09 13:08:07 -05:00
|
|
|
let url = ~"http://rust-lang.org#something";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-09 13:08:07 -05:00
|
|
|
}
|
|
|
|
|
2012-07-24 22:21:32 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_url_component_encoding() {
|
2012-07-24 22:21:32 -05:00
|
|
|
let url = ~"http://rust-lang.org/doc%20uments?ba%25d%20=%23%26%2B";
|
2012-12-21 09:47:32 -06:00
|
|
|
let u = from_str(url).unwrap();
|
2012-07-24 22:21:32 -05:00
|
|
|
assert u.path == ~"/doc uments";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert u.query == ~[(~"ba%d ", ~"#&+")];
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
2012-07-31 23:40:38 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_url_without_authority() {
|
2012-07-31 23:40:38 -05:00
|
|
|
let url = ~"mailto:test@email.com";
|
2012-12-21 09:47:32 -06:00
|
|
|
assert from_str(url).unwrap().to_str() == url;
|
2012-07-31 23:40:38 -05:00
|
|
|
}
|
|
|
|
|
2012-07-24 22:21:32 -05:00
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_encode() {
|
2012-09-03 14:33:51 -05:00
|
|
|
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("]") == ~"]";
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_encode_component() {
|
2012-12-21 09:47:32 -06:00
|
|
|
assert encode_component("") == ~"";
|
|
|
|
assert encode_component("http://example.com") ==
|
2012-07-24 22:21:32 -05:00
|
|
|
~"http%3A%2F%2Fexample.com";
|
2012-12-21 09:47:32 -06:00
|
|
|
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";
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_decode() {
|
2012-12-21 09:47:32 -06:00
|
|
|
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";
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_decode_component() {
|
2012-12-21 09:47:32 -06:00
|
|
|
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") == ~"]";
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_encode_form_urlencoded() {
|
2013-01-23 16:06:32 -06:00
|
|
|
let mut m = LinearMap::new();
|
2012-12-21 09:47:32 -06:00
|
|
|
assert encode_form_urlencoded(&m) == ~"";
|
2012-07-24 22:21:32 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
m.insert(~"", ~[]);
|
|
|
|
m.insert(~"foo", ~[]);
|
|
|
|
assert encode_form_urlencoded(&m) == ~"";
|
2012-07-24 22:21:32 -05:00
|
|
|
|
2013-01-23 16:06:32 -06:00
|
|
|
let mut m = LinearMap::new();
|
2012-12-21 09:47:32 -06:00
|
|
|
m.insert(~"foo", ~[~"bar", ~"123"]);
|
|
|
|
assert encode_form_urlencoded(&m) == ~"foo=bar&foo=123";
|
2012-07-24 22:21:32 -05:00
|
|
|
|
2013-01-23 16:06:32 -06:00
|
|
|
let mut m = LinearMap::new();
|
2012-12-21 09:47:32 -06:00
|
|
|
m.insert(~"foo bar", ~[~"abc", ~"12 = 34"]);
|
|
|
|
assert encode_form_urlencoded(&m) == ~"foo+bar=abc&foo+bar=12+%3D+34";
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-01-29 14:06:09 -06:00
|
|
|
pub fn test_decode_form_urlencoded() {
|
2013-01-11 15:40:15 -06:00
|
|
|
// FIXME #4449: Commented out because this causes an ICE, but only
|
|
|
|
// on FreeBSD
|
|
|
|
/*
|
2013-01-10 22:08:49 -06:00
|
|
|
assert decode_form_urlencoded(~[]).len() == 0;
|
2012-07-24 22:21:32 -05:00
|
|
|
|
2012-12-21 09:47:32 -06:00
|
|
|
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"];
|
2013-01-10 22:08:49 -06:00
|
|
|
assert form.get_ref(&~"foo bar") == &~[~"abc", ~"12 = 34"];
|
2013-01-11 15:40:15 -06:00
|
|
|
*/
|
2012-07-24 22:21:32 -05:00
|
|
|
}
|
2012-07-28 00:23:36 -05:00
|
|
|
}
|