rust/src/libstd/net_ip.rs

390 lines
12 KiB
Rust
Raw Normal View History

//! Types/fns concerning Internet Protocol (IP), versions 4 & 6
2012-09-02 22:18:08 -07:00
#[forbid(deprecated_mode)];
#[forbid(deprecated_pattern)];
2012-09-04 11:23:53 -07:00
use iotask = uv::iotask::IoTask;
use interact = uv::iotask::interact;
2012-09-04 11:23:53 -07:00
use sockaddr_in = uv::ll::sockaddr_in;
use sockaddr_in6 = uv::ll::sockaddr_in6;
use addrinfo = uv::ll::addrinfo;
use uv_getaddrinfo_t = uv::ll::uv_getaddrinfo_t;
use uv_ip4_addr = uv::ll::ip4_addr;
use uv_ip4_name = uv::ll::ip4_name;
use uv_ip6_addr = uv::ll::ip6_addr;
use uv_ip6_name = uv::ll::ip6_name;
use uv_getaddrinfo = uv::ll::getaddrinfo;
use uv_freeaddrinfo = uv::ll::freeaddrinfo;
use create_uv_getaddrinfo_t = uv::ll::getaddrinfo_t;
use set_data_for_req = uv::ll::set_data_for_req;
use get_data_for_req = uv::ll::get_data_for_req;
use ll = uv::ll;
use comm = core::comm;
2012-08-30 11:01:39 -07:00
export IpAddr, parse_addr_err;
2012-04-30 15:10:24 -07:00
export format_addr;
export v4, v6;
export get_addr;
2012-08-30 11:01:39 -07:00
export Ipv4, Ipv6;
/// An IP address
2012-08-30 11:01:39 -07:00
enum IpAddr {
/// An IPv4 address
2012-08-30 11:01:39 -07:00
Ipv4(sockaddr_in),
Ipv6(sockaddr_in6)
}
/// Human-friendly feedback on why a parse_addr attempt failed
2012-08-30 11:01:39 -07:00
type ParseAddrErr = {
err_msg: ~str
};
/**
* Convert a `ip_addr` to a str
*
* # Arguments
*
* * ip - a `std::net::ip::ip_addr`
*/
2012-09-02 22:18:08 -07:00
fn format_addr(ip: &IpAddr) -> ~str {
match *ip {
2012-08-30 11:01:39 -07:00
Ipv4(addr) => unsafe {
2012-08-03 19:59:04 -07:00
let result = uv_ip4_name(&addr);
if result == ~"" {
fail ~"failed to convert inner sockaddr_in address to str"
}
2012-08-03 19:59:04 -07:00
result
},
2012-08-30 11:01:39 -07:00
Ipv6(addr) => unsafe {
2012-08-03 19:59:04 -07:00
let result = uv_ip6_name(&addr);
if result == ~"" {
fail ~"failed to convert inner sockaddr_in address to str"
}
2012-08-03 19:59:04 -07:00
result
}
}
}
/// Represents errors returned from `net::ip::get_addr()`
2012-08-30 11:01:39 -07:00
enum IpGetAddrErr {
GetAddrUnknownError
}
/**
* Attempts name resolution on the provided `node` string
*
* # Arguments
*
* * `node` - a string representing some host address
* * `iotask` - a `uv::iotask` used to interact with the underlying event loop
*
* # Returns
*
* A `result<~[ip_addr], ip_get_addr_err>` instance that will contain
* a vector of `ip_addr` results, in the case of success, or an error
* object in the case of failure
*/
2012-09-02 22:18:08 -07:00
fn get_addr(node: &str, iotask: iotask)
2012-08-30 11:01:39 -07:00
-> result::Result<~[IpAddr], IpGetAddrErr> {
2012-08-14 14:17:27 -07:00
do core::comm::listen |output_ch| {
do str::as_buf(node) |node_ptr, len| unsafe {
2012-08-22 17:24:52 -07:00
log(debug, fmt!("slice len %?", len));
let handle = create_uv_getaddrinfo_t();
let handle_ptr = ptr::addr_of(handle);
2012-08-30 11:01:39 -07:00
let handle_data: GetAddrData = {
output_ch: output_ch
};
let handle_data_ptr = ptr::addr_of(handle_data);
do interact(iotask) |loop_ptr| unsafe {
let result = uv_getaddrinfo(
loop_ptr,
handle_ptr,
get_addr_cb,
node_ptr,
ptr::null(),
ptr::null());
2012-08-06 12:34:08 -07:00
match result {
2012-08-03 19:59:04 -07:00
0i32 => {
set_data_for_req(handle_ptr, handle_data_ptr);
}
2012-08-03 19:59:04 -07:00
_ => {
2012-08-30 11:01:39 -07:00
output_ch.send(result::Err(GetAddrUnknownError));
}
}
};
output_ch.recv()
}
}
}
mod v4 {
#[legacy_exports];
/**
* Convert a str to `ip_addr`
*
* # Failure
*
* Fails if the string is not a valid IPv4 address
*
* # Arguments
*
* * ip - a string of the format `x.x.x.x`
*
* # Returns
*
* * an `ip_addr` of the `ipv4` variant
*/
2012-09-02 22:18:08 -07:00
fn parse_addr(ip: &str) -> IpAddr {
2012-08-06 12:34:08 -07:00
match try_parse_addr(ip) {
2012-08-26 16:54:31 -07:00
result::Ok(addr) => copy(addr),
result::Err(err_data) => fail err_data.err_msg
}
}
// the simple, old style numberic representation of
// ipv4
2012-08-30 11:01:39 -07:00
type Ipv4Rep = { a: u8, b: u8, c: u8, d:u8 };
2012-08-30 11:01:39 -07:00
trait AsUnsafeU32 {
unsafe fn as_u32() -> u32;
}
2012-08-30 11:01:39 -07:00
impl Ipv4Rep: AsUnsafeU32 {
// this is pretty dastardly, i know
unsafe fn as_u32() -> u32 {
*((ptr::addr_of(self)) as *u32)
}
}
2012-09-02 22:18:08 -07:00
fn parse_to_ipv4_rep(ip: &str) -> result::Result<Ipv4Rep, ~str> {
2012-06-30 16:19:07 -07:00
let parts = vec::map(str::split_char(ip, '.'), |s| {
2012-08-06 12:34:08 -07:00
match uint::from_str(s) {
2012-08-20 12:23:37 -07:00
Some(n) if n <= 255u => n,
2012-08-03 19:59:04 -07:00
_ => 256u
}
});
if vec::len(parts) != 4u {
2012-08-26 16:54:31 -07:00
result::Err(fmt!("'%s' doesn't have 4 parts", ip))
}
else if vec::contains(parts, 256u) {
2012-08-26 16:54:31 -07:00
result::Err(fmt!("invalid octal in addr '%s'", ip))
}
else {
2012-08-26 16:54:31 -07:00
result::Ok({a: parts[0] as u8, b: parts[1] as u8,
c: parts[2] as u8, d: parts[3] as u8})
}
}
2012-09-02 22:18:08 -07:00
fn try_parse_addr(ip: &str) -> result::Result<IpAddr,ParseAddrErr> {
unsafe {
let INADDR_NONE = ll::get_INADDR_NONE();
let ip_rep_result = parse_to_ipv4_rep(ip);
if result::is_err(ip_rep_result) {
let err_str = result::get_err(ip_rep_result);
2012-08-26 16:54:31 -07:00
return result::Err({err_msg: err_str})
}
// ipv4_rep.as_u32 is unsafe :/
let input_is_inaddr_none =
result::get(ip_rep_result).as_u32() == INADDR_NONE;
2012-09-02 22:18:08 -07:00
let new_addr = uv_ip4_addr(str::from_slice(ip), 22);
let reformatted_name = uv_ip4_name(&new_addr);
2012-08-22 17:24:52 -07:00
log(debug, fmt!("try_parse_addr: input ip: %s reparsed ip: %s",
ip, reformatted_name));
let ref_ip_rep_result = parse_to_ipv4_rep(reformatted_name);
if result::is_err(ref_ip_rep_result) {
let err_str = result::get_err(ref_ip_rep_result);
2012-08-26 16:54:31 -07:00
return result::Err({err_msg: err_str})
}
if result::get(ref_ip_rep_result).as_u32() == INADDR_NONE &&
!input_is_inaddr_none {
2012-08-26 16:54:31 -07:00
return result::Err(
{err_msg: ~"uv_ip4_name produced invalid result."})
}
else {
2012-08-30 11:01:39 -07:00
result::Ok(Ipv4(copy(new_addr)))
}
}
}
}
mod v6 {
#[legacy_exports];
/**
* Convert a str to `ip_addr`
*
* # Failure
*
* Fails if the string is not a valid IPv6 address
*
* # Arguments
*
* * ip - an ipv6 string. See RFC2460 for spec.
*
* # Returns
*
* * an `ip_addr` of the `ipv6` variant
*/
2012-09-02 22:18:08 -07:00
fn parse_addr(ip: &str) -> IpAddr {
2012-08-06 12:34:08 -07:00
match try_parse_addr(ip) {
2012-08-26 16:54:31 -07:00
result::Ok(addr) => copy(addr),
result::Err(err_data) => fail err_data.err_msg
}
}
2012-09-02 22:18:08 -07:00
fn try_parse_addr(ip: &str) -> result::Result<IpAddr,ParseAddrErr> {
unsafe {
// need to figure out how to establish a parse failure..
2012-09-02 22:18:08 -07:00
let new_addr = uv_ip6_addr(str::from_slice(ip), 22);
let reparsed_name = uv_ip6_name(&new_addr);
2012-08-22 17:24:52 -07:00
log(debug, fmt!("v6::try_parse_addr ip: '%s' reparsed '%s'",
ip, reparsed_name));
// '::' appears to be uv_ip6_name() returns for bogus
// parses..
2012-09-02 22:18:08 -07:00
if ip != &"::" && reparsed_name == ~"::" {
2012-08-26 16:54:31 -07:00
result::Err({err_msg:fmt!("failed to parse '%s'",
2012-08-22 17:24:52 -07:00
ip)})
}
else {
2012-08-30 11:01:39 -07:00
result::Ok(Ipv6(new_addr))
}
}
}
}
2012-08-30 11:01:39 -07:00
type GetAddrData = {
output_ch: comm::Chan<result::Result<~[IpAddr],IpGetAddrErr>>
};
2012-07-03 16:32:02 -07:00
extern fn get_addr_cb(handle: *uv_getaddrinfo_t, status: libc::c_int,
res: *addrinfo) unsafe {
log(debug, ~"in get_addr_cb");
let handle_data = get_data_for_req(handle) as
2012-08-30 11:01:39 -07:00
*GetAddrData;
if status == 0i32 {
if res != (ptr::null::<addrinfo>()) {
let mut out_vec = ~[];
2012-08-22 17:24:52 -07:00
log(debug, fmt!("initial addrinfo: %?", res));
let mut curr_addr = res;
loop {
let new_ip_addr = if ll::is_ipv4_addrinfo(curr_addr) {
2012-08-30 11:01:39 -07:00
Ipv4(copy((
*ll::addrinfo_as_sockaddr_in(curr_addr))))
}
else if ll::is_ipv6_addrinfo(curr_addr) {
2012-08-30 11:01:39 -07:00
Ipv6(copy((
*ll::addrinfo_as_sockaddr_in6(curr_addr))))
}
else {
log(debug, ~"curr_addr is not of family AF_INET or "+
~"AF_INET6. Error.");
(*handle_data).output_ch.send(
2012-08-30 11:01:39 -07:00
result::Err(GetAddrUnknownError));
break;
};
vec::push(out_vec, move new_ip_addr);
let next_addr = ll::get_next_addrinfo(curr_addr);
if next_addr == ptr::null::<addrinfo>() as *addrinfo {
log(debug, ~"null next_addr encountered. no mas");
break;
}
else {
curr_addr = next_addr;
2012-08-22 17:24:52 -07:00
log(debug, fmt!("next_addr addrinfo: %?", curr_addr));
}
}
2012-08-22 17:24:52 -07:00
log(debug, fmt!("successful process addrinfo result, len: %?",
vec::len(out_vec)));
2012-09-11 17:17:54 -07:00
(*handle_data).output_ch.send(result::Ok(move out_vec));
}
else {
log(debug, ~"addrinfo pointer is NULL");
(*handle_data).output_ch.send(
2012-08-30 11:01:39 -07:00
result::Err(GetAddrUnknownError));
}
}
else {
log(debug, ~"status != 0 error in get_addr_cb");
(*handle_data).output_ch.send(
2012-08-30 11:01:39 -07:00
result::Err(GetAddrUnknownError));
}
if res != (ptr::null::<addrinfo>()) {
uv_freeaddrinfo(res);
}
log(debug, ~"leaving get_addr_cb");
}
#[cfg(test)]
mod test {
#[legacy_exports];
#[test]
2012-06-26 16:13:50 -07:00
fn test_ip_ipv4_parse_and_format_ip() {
let localhost_str = ~"127.0.0.1";
2012-09-02 22:18:08 -07:00
assert (format_addr(&v4::parse_addr(localhost_str))
== localhost_str)
}
#[test]
2012-06-26 16:13:50 -07:00
fn test_ip_ipv6_parse_and_format_ip() {
let localhost_str = ~"::1";
2012-09-02 22:18:08 -07:00
let format_result = format_addr(&v6::parse_addr(localhost_str));
2012-08-22 17:24:52 -07:00
log(debug, fmt!("results: expected: '%s' actual: '%s'",
localhost_str, format_result));
assert format_result == localhost_str;
}
#[test]
2012-06-26 16:13:50 -07:00
fn test_ip_ipv4_bad_parse() {
2012-08-06 12:34:08 -07:00
match v4::try_parse_addr(~"b4df00d") {
2012-08-26 16:54:31 -07:00
result::Err(err_info) => {
2012-08-22 17:24:52 -07:00
log(debug, fmt!("got error as expected %?", err_info));
assert true;
}
2012-08-26 16:54:31 -07:00
result::Ok(addr) => {
2012-08-22 17:24:52 -07:00
fail fmt!("Expected failure, but got addr %?", addr);
}
}
}
#[test]
#[ignore(target_os="win32")]
2012-06-26 16:13:50 -07:00
fn test_ip_ipv6_bad_parse() {
2012-08-06 12:34:08 -07:00
match v6::try_parse_addr(~"::,~2234k;") {
2012-08-26 16:54:31 -07:00
result::Err(err_info) => {
2012-08-22 17:24:52 -07:00
log(debug, fmt!("got error as expected %?", err_info));
assert true;
}
2012-08-26 16:54:31 -07:00
result::Ok(addr) => {
2012-08-22 17:24:52 -07:00
fail fmt!("Expected failure, but got addr %?", addr);
}
}
}
#[test]
2012-06-29 15:35:47 -07:00
#[ignore(reason = "valgrind says it's leaky")]
2012-06-26 16:13:50 -07:00
fn test_ip_get_addr() {
let localhost_name = ~"localhost";
let iotask = uv::global_loop::get();
let ga_result = get_addr(localhost_name, iotask);
if result::is_err(ga_result) {
fail ~"got err result from net::ip::get_addr();"
}
// note really sure how to realiably test/assert
// this.. mostly just wanting to see it work, atm.
let results = result::unwrap(ga_result);
2012-08-22 17:24:52 -07:00
log(debug, fmt!("test_get_addr: Number of results for %s: %?",
localhost_name, vec::len(results)));
for vec::each(results) |r| {
let ipv_prefix = match *r {
2012-08-30 11:01:39 -07:00
Ipv4(_) => ~"IPv4",
Ipv6(_) => ~"IPv6"
};
2012-08-22 17:24:52 -07:00
log(debug, fmt!("test_get_addr: result %s: '%s'",
ipv_prefix, format_addr(r)));
}
// at least one result.. this is going to vary from system
// to system, based on stuff like the contents of /etc/hosts
assert vec::len(results) > 0;
}
#[test]
2012-06-29 15:35:47 -07:00
#[ignore(reason = "valgrind says it's leaky")]
2012-06-26 16:13:50 -07:00
fn test_ip_get_addr_bad_input() {
let localhost_name = ~"sjkl234m,./sdf";
let iotask = uv::global_loop::get();
let ga_result = get_addr(localhost_name, iotask);
assert result::is_err(ga_result);
}
}