2012-12-03 16:48:01 -08: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-04 22:53:12 +01:00
|
|
|
//! Types/fns concerning Internet Protocol (IP), versions 4 & 6
|
2012-09-02 22:18:08 -07:00
|
|
|
#[forbid(deprecated_mode)];
|
2012-04-29 21:53:17 -07:00
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use core::libc;
|
2013-01-08 19:37:25 -08:00
|
|
|
use core::prelude::*;
|
2013-01-25 00:52:50 -08:00
|
|
|
use core::pipes::{stream, SharedChan};
|
2012-12-23 17:41:37 -05:00
|
|
|
use core::ptr;
|
|
|
|
use core::result;
|
|
|
|
use core::str;
|
|
|
|
use core::uint;
|
|
|
|
use core::vec;
|
|
|
|
|
2012-09-04 11:23:53 -07:00
|
|
|
use iotask = uv::iotask::IoTask;
|
|
|
|
use interact = uv::iotask::interact;
|
2012-04-29 21:53:17 -07:00
|
|
|
|
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;
|
2012-10-19 19:28:58 -04:00
|
|
|
use uv_ip4_port = uv::ll::ip4_port;
|
2012-09-04 11:23:53 -07:00
|
|
|
use uv_ip6_addr = uv::ll::ip6_addr;
|
|
|
|
use uv_ip6_name = uv::ll::ip6_name;
|
2012-10-19 19:28:58 -04:00
|
|
|
use uv_ip6_port = uv::ll::ip6_port;
|
2012-09-04 11:23:53 -07:00
|
|
|
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;
|
2012-04-29 21:53:17 -07:00
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/// An IP address
|
2012-10-03 16:43:56 -07:00
|
|
|
pub enum IpAddr {
|
2012-07-04 22:53:12 +01:00
|
|
|
/// An IPv4 address
|
2012-08-30 11:01:39 -07:00
|
|
|
Ipv4(sockaddr_in),
|
|
|
|
Ipv6(sockaddr_in6)
|
2012-04-29 21:53:17 -07:00
|
|
|
}
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/// Human-friendly feedback on why a parse_addr attempt failed
|
2013-01-22 08:44:24 -08:00
|
|
|
pub struct ParseAddrErr {
|
|
|
|
err_msg: ~str,
|
|
|
|
}
|
2012-05-19 14:06:48 -07:00
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/**
|
2012-10-19 19:28:58 -04:00
|
|
|
* Convert a `IpAddr` to a str
|
2012-07-04 22:53:12 +01:00
|
|
|
*
|
|
|
|
* # Arguments
|
|
|
|
*
|
2012-10-19 19:28:58 -04:00
|
|
|
* * ip - a `std::net::ip::IpAddr`
|
2012-07-04 22:53:12 +01:00
|
|
|
*/
|
2012-10-03 16:43:56 -07:00
|
|
|
pub fn format_addr(ip: &IpAddr) -> ~str {
|
2012-09-02 22:18:08 -07:00
|
|
|
match *ip {
|
2012-09-28 00:22:18 -07:00
|
|
|
Ipv4(ref addr) => unsafe {
|
|
|
|
let result = uv_ip4_name(addr);
|
2012-08-03 19:59:04 -07:00
|
|
|
if result == ~"" {
|
|
|
|
fail ~"failed to convert inner sockaddr_in address to str"
|
2012-06-17 20:36:36 -07:00
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
result
|
2012-08-06 17:14:32 -07:00
|
|
|
},
|
2012-09-28 00:22:18 -07:00
|
|
|
Ipv6(ref addr) => unsafe {
|
|
|
|
let result = uv_ip6_name(addr);
|
2012-08-03 19:59:04 -07:00
|
|
|
if result == ~"" {
|
|
|
|
fail ~"failed to convert inner sockaddr_in address to str"
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
result
|
2012-05-05 09:21:19 -07:00
|
|
|
}
|
2012-04-29 21:53:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-19 19:28:58 -04:00
|
|
|
/**
|
|
|
|
* Get the associated port
|
|
|
|
*
|
|
|
|
* # Arguments
|
|
|
|
* * ip - a `std::net::ip::IpAddr`
|
|
|
|
*/
|
|
|
|
pub fn get_port(ip: &IpAddr) -> uint {
|
|
|
|
match *ip {
|
|
|
|
Ipv4(ref addr) => unsafe {
|
|
|
|
uv_ip4_port(addr)
|
|
|
|
},
|
|
|
|
Ipv6(ref addr) => unsafe {
|
|
|
|
uv_ip6_port(addr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/// Represents errors returned from `net::ip::get_addr()`
|
2012-08-30 11:01:39 -07:00
|
|
|
enum IpGetAddrErr {
|
|
|
|
GetAddrUnknownError
|
2012-06-22 15:31:56 -07:00
|
|
|
}
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*
|
2012-07-11 16:49:02 -07:00
|
|
|
* A `result<~[ip_addr], ip_get_addr_err>` instance that will contain
|
2012-07-04 22:53:12 +01:00
|
|
|
* a vector of `ip_addr` results, in the case of success, or an error
|
|
|
|
* object in the case of failure
|
2013-01-25 00:52:50 -08:00
|
|
|
*/
|
2013-01-19 23:38:17 -08:00
|
|
|
pub fn get_addr(node: &str, iotask: &iotask)
|
2013-01-25 00:52:50 -08:00
|
|
|
-> result::Result<~[IpAddr], IpGetAddrErr> {
|
|
|
|
let (output_po, output_ch) = stream();
|
2013-01-10 10:59:58 -08:00
|
|
|
let mut output_ch = Some(SharedChan(output_ch));
|
2013-01-25 00:52:50 -08:00
|
|
|
do str::as_buf(node) |node_ptr, len| {
|
2013-01-10 10:59:58 -08:00
|
|
|
let output_ch = output_ch.swap_unwrap();
|
2013-01-25 00:52:50 -08:00
|
|
|
unsafe {
|
|
|
|
log(debug, fmt!("slice len %?", len));
|
|
|
|
let handle = create_uv_getaddrinfo_t();
|
|
|
|
let handle_ptr = ptr::addr_of(&handle);
|
|
|
|
let handle_data = GetAddrData {
|
|
|
|
output_ch: output_ch.clone()
|
|
|
|
};
|
|
|
|
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());
|
|
|
|
match result {
|
|
|
|
0i32 => {
|
2013-01-23 11:43:58 -08:00
|
|
|
set_data_for_req(handle_ptr, handle_data_ptr);
|
2013-01-25 00:52:50 -08:00
|
|
|
}
|
|
|
|
_ => {
|
2013-01-23 11:43:58 -08:00
|
|
|
output_ch.send(result::Err(GetAddrUnknownError));
|
|
|
|
}
|
|
|
|
}
|
2013-01-25 00:52:50 -08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
output_po.recv()
|
2012-06-22 15:31:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-03 16:43:56 -07:00
|
|
|
pub mod v4 {
|
2013-01-08 19:37:25 -08:00
|
|
|
use net::ip::{IpAddr, Ipv4, Ipv6, ParseAddrErr};
|
2012-12-23 17:41:37 -05:00
|
|
|
use uv::ll;
|
2013-01-08 19:37:25 -08:00
|
|
|
use uv_ip4_addr = uv::ll::ip4_addr;
|
|
|
|
use uv_ip4_name = uv::ll::ip4_name;
|
2012-12-23 17:41:37 -05:00
|
|
|
|
2013-01-08 19:37:25 -08:00
|
|
|
use core::prelude::*;
|
2012-12-23 17:41:37 -05:00
|
|
|
use core::ptr;
|
|
|
|
use core::result;
|
|
|
|
use core::str;
|
|
|
|
use core::uint;
|
|
|
|
use core::vec;
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/**
|
|
|
|
* 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-10-03 16:43:56 -07:00
|
|
|
pub fn parse_addr(ip: &str) -> IpAddr {
|
2012-08-06 12:34:08 -07:00
|
|
|
match try_parse_addr(ip) {
|
2012-09-18 22:35:42 -07:00
|
|
|
result::Ok(move addr) => move addr,
|
2012-09-28 00:22:18 -07:00
|
|
|
result::Err(ref err_data) => fail err_data.err_msg
|
2012-05-19 14:06:48 -07:00
|
|
|
}
|
|
|
|
}
|
2012-06-26 09:07:24 -07:00
|
|
|
// the simple, old style numberic representation of
|
|
|
|
// ipv4
|
2013-01-22 08:44:24 -08:00
|
|
|
pub struct Ipv4Rep { a: u8, b: u8, c: u8, d: u8 }
|
2012-07-11 15:00:40 -07:00
|
|
|
|
2012-10-03 16:43:56 -07:00
|
|
|
pub trait AsUnsafeU32 {
|
2012-07-11 15:00:40 -07:00
|
|
|
unsafe fn as_u32() -> u32;
|
|
|
|
}
|
|
|
|
|
2012-08-30 11:01:39 -07:00
|
|
|
impl Ipv4Rep: AsUnsafeU32 {
|
2012-06-26 09:07:24 -07:00
|
|
|
// this is pretty dastardly, i know
|
|
|
|
unsafe fn as_u32() -> u32 {
|
2012-10-01 12:47:02 -07:00
|
|
|
*((ptr::addr_of(&self)) as *u32)
|
2012-06-26 09:07:24 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-03 16:43:56 -07:00
|
|
|
pub 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-09-21 18:43:30 -07:00
|
|
|
match uint::from_str(*s) {
|
2012-09-28 12:22:33 -07:00
|
|
|
Some(n) if n <= 255 => n,
|
|
|
|
_ => 256
|
2012-04-29 21:53:17 -07:00
|
|
|
}
|
|
|
|
});
|
2012-09-28 12:22:33 -07:00
|
|
|
if parts.len() != 4 {
|
2013-01-22 08:44:24 -08:00
|
|
|
Err(fmt!("'%s' doesn't have 4 parts", ip))
|
|
|
|
} else if parts.contains(&256) {
|
|
|
|
Err(fmt!("invalid octal in addr '%s'", ip))
|
|
|
|
} else {
|
|
|
|
Ok(Ipv4Rep {
|
|
|
|
a: parts[0] as u8, b: parts[1] as u8,
|
|
|
|
c: parts[2] as u8, d: parts[3] as u8,
|
|
|
|
})
|
2012-06-26 09:07:24 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-03 16:43:56 -07:00
|
|
|
pub fn try_parse_addr(ip: &str) -> result::Result<IpAddr,ParseAddrErr> {
|
2012-06-17 20:36:36 -07:00
|
|
|
unsafe {
|
2012-06-26 09:07:24 -07:00
|
|
|
let INADDR_NONE = ll::get_INADDR_NONE();
|
2012-06-26 12:47:44 -07:00
|
|
|
let ip_rep_result = parse_to_ipv4_rep(ip);
|
2012-09-25 16:23:04 -07:00
|
|
|
if result::is_err(&ip_rep_result) {
|
|
|
|
let err_str = result::get_err(&ip_rep_result);
|
2013-01-22 08:44:24 -08:00
|
|
|
return result::Err(ParseAddrErr { err_msg: err_str })
|
2012-06-26 09:07:24 -07:00
|
|
|
}
|
|
|
|
// ipv4_rep.as_u32 is unsafe :/
|
|
|
|
let input_is_inaddr_none =
|
2012-09-25 16:23:04 -07:00
|
|
|
result::get(&ip_rep_result).as_u32() == INADDR_NONE;
|
2012-06-26 09:07:24 -07:00
|
|
|
|
2012-09-02 22:18:08 -07:00
|
|
|
let new_addr = uv_ip4_addr(str::from_slice(ip), 22);
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
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));
|
2012-06-26 12:47:44 -07:00
|
|
|
let ref_ip_rep_result = parse_to_ipv4_rep(reformatted_name);
|
2012-09-25 16:23:04 -07:00
|
|
|
if result::is_err(&ref_ip_rep_result) {
|
|
|
|
let err_str = result::get_err(&ref_ip_rep_result);
|
2013-01-22 08:44:24 -08:00
|
|
|
return Err(ParseAddrErr { err_msg: err_str })
|
2012-06-26 09:07:24 -07:00
|
|
|
}
|
2013-01-22 08:44:24 -08:00
|
|
|
|
2012-09-25 16:23:04 -07:00
|
|
|
if result::get(&ref_ip_rep_result).as_u32() == INADDR_NONE &&
|
2012-06-26 09:07:24 -07:00
|
|
|
!input_is_inaddr_none {
|
2013-01-22 08:44:24 -08:00
|
|
|
Err(ParseAddrErr {
|
|
|
|
err_msg: ~"uv_ip4_name produced invalid result.",
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
Ok(Ipv4(copy(new_addr)))
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
}
|
2012-05-19 14:06:48 -07:00
|
|
|
}
|
2012-04-29 21:53:17 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-03 16:43:56 -07:00
|
|
|
pub mod v6 {
|
2013-01-08 19:37:25 -08:00
|
|
|
use net::ip::{IpAddr, Ipv6, ParseAddrErr};
|
|
|
|
use uv_ip6_addr = uv::ll::ip6_addr;
|
|
|
|
use uv_ip6_name = uv::ll::ip6_name;
|
|
|
|
|
|
|
|
use core::prelude::*;
|
2012-12-23 17:41:37 -05:00
|
|
|
use core::result;
|
|
|
|
use core::str;
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
/**
|
|
|
|
* 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-10-03 16:43:56 -07:00
|
|
|
pub fn parse_addr(ip: &str) -> IpAddr {
|
2012-08-06 12:34:08 -07:00
|
|
|
match try_parse_addr(ip) {
|
2012-09-18 22:35:42 -07:00
|
|
|
result::Ok(move addr) => move addr,
|
2012-09-28 00:22:18 -07:00
|
|
|
result::Err(copy err_data) => fail err_data.err_msg
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-03 16:43:56 -07:00
|
|
|
pub fn try_parse_addr(ip: &str) -> result::Result<IpAddr,ParseAddrErr> {
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
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);
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
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));
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
// '::' appears to be uv_ip6_name() returns for bogus
|
|
|
|
// parses..
|
2012-09-02 22:18:08 -07:00
|
|
|
if ip != &"::" && reparsed_name == ~"::" {
|
2013-01-22 08:44:24 -08:00
|
|
|
Err(ParseAddrErr { err_msg:fmt!("failed to parse '%s'", ip) })
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
}
|
|
|
|
else {
|
2013-01-22 08:44:24 -08:00
|
|
|
Ok(Ipv6(new_addr))
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-22 08:44:24 -08:00
|
|
|
struct GetAddrData {
|
2013-01-25 00:52:50 -08:00
|
|
|
output_ch: SharedChan<result::Result<~[IpAddr],IpGetAddrErr>>
|
2013-01-22 08:44:24 -08:00
|
|
|
}
|
2012-06-27 15:59:04 -07:00
|
|
|
|
2012-07-03 16:32:02 -07:00
|
|
|
extern fn get_addr_cb(handle: *uv_getaddrinfo_t, status: libc::c_int,
|
2013-01-23 11:43:58 -08:00
|
|
|
res: *addrinfo) {
|
|
|
|
unsafe {
|
|
|
|
log(debug, ~"in get_addr_cb");
|
|
|
|
let handle_data = get_data_for_req(handle) as
|
|
|
|
*GetAddrData;
|
2013-01-25 00:52:50 -08:00
|
|
|
let output_ch = (*handle_data).output_ch.clone();
|
2013-01-23 11:43:58 -08:00
|
|
|
if status == 0i32 {
|
|
|
|
if res != (ptr::null::<addrinfo>()) {
|
|
|
|
let mut out_vec = ~[];
|
|
|
|
log(debug, fmt!("initial addrinfo: %?", res));
|
|
|
|
let mut curr_addr = res;
|
|
|
|
loop {
|
|
|
|
let new_ip_addr = if ll::is_ipv4_addrinfo(curr_addr) {
|
|
|
|
Ipv4(copy((
|
|
|
|
*ll::addrinfo_as_sockaddr_in(curr_addr))))
|
|
|
|
}
|
|
|
|
else if ll::is_ipv6_addrinfo(curr_addr) {
|
|
|
|
Ipv6(copy((
|
|
|
|
*ll::addrinfo_as_sockaddr_in6(curr_addr))))
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log(debug, ~"curr_addr is not of family AF_INET or "+
|
|
|
|
~"AF_INET6. Error.");
|
2013-01-25 00:52:50 -08:00
|
|
|
output_ch.send(
|
2013-01-23 11:43:58 -08:00
|
|
|
result::Err(GetAddrUnknownError));
|
|
|
|
break;
|
|
|
|
};
|
|
|
|
out_vec.push(move new_ip_addr);
|
2012-06-27 15:59:04 -07:00
|
|
|
|
2013-01-23 11:43:58 -08:00
|
|
|
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;
|
|
|
|
log(debug, fmt!("next_addr addrinfo: %?", curr_addr));
|
|
|
|
}
|
2012-06-27 15:59:04 -07:00
|
|
|
}
|
2013-01-23 11:43:58 -08:00
|
|
|
log(debug, fmt!("successful process addrinfo result, len: %?",
|
|
|
|
vec::len(out_vec)));
|
2013-01-25 00:52:50 -08:00
|
|
|
output_ch.send(result::Ok(move out_vec));
|
2013-01-23 11:43:58 -08:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
log(debug, ~"addrinfo pointer is NULL");
|
2013-01-25 00:52:50 -08:00
|
|
|
output_ch.send(
|
2013-01-23 11:43:58 -08:00
|
|
|
result::Err(GetAddrUnknownError));
|
2012-06-27 15:59:04 -07:00
|
|
|
}
|
2012-05-19 14:06:48 -07:00
|
|
|
}
|
|
|
|
else {
|
2013-01-23 11:43:58 -08:00
|
|
|
log(debug, ~"status != 0 error in get_addr_cb");
|
2013-01-25 00:52:50 -08:00
|
|
|
output_ch.send(
|
2012-08-30 11:01:39 -07:00
|
|
|
result::Err(GetAddrUnknownError));
|
2012-05-19 14:06:48 -07:00
|
|
|
}
|
2013-01-23 11:43:58 -08:00
|
|
|
if res != (ptr::null::<addrinfo>()) {
|
|
|
|
uv_freeaddrinfo(res);
|
|
|
|
}
|
|
|
|
log(debug, ~"leaving get_addr_cb");
|
2012-04-29 21:53:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
2013-01-08 19:37:25 -08:00
|
|
|
use core::prelude::*;
|
|
|
|
|
|
|
|
use net_ip::*;
|
2012-12-28 12:46:08 -08:00
|
|
|
use net_ip::v4;
|
|
|
|
use net_ip::v6;
|
|
|
|
use uv;
|
|
|
|
|
|
|
|
use core::result;
|
|
|
|
use core::vec;
|
|
|
|
|
2012-04-29 21:53:17 -07:00
|
|
|
#[test]
|
2012-06-26 16:13:50 -07:00
|
|
|
fn test_ip_ipv4_parse_and_format_ip() {
|
2012-07-13 22:57:48 -07:00
|
|
|
let localhost_str = ~"127.0.0.1";
|
2012-09-02 22:18:08 -07:00
|
|
|
assert (format_addr(&v4::parse_addr(localhost_str))
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
== localhost_str)
|
|
|
|
}
|
|
|
|
#[test]
|
2012-06-26 16:13:50 -07:00
|
|
|
fn test_ip_ipv6_parse_and_format_ip() {
|
2012-07-13 22:57:48 -07:00
|
|
|
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));
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
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-09-28 02:26:20 -07:00
|
|
|
result::Err(ref err_info) => {
|
2012-08-22 17:24:52 -07:00
|
|
|
log(debug, fmt!("got error as expected %?", err_info));
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
assert true;
|
|
|
|
}
|
2012-09-28 02:26:20 -07:00
|
|
|
result::Ok(ref addr) => {
|
2012-08-22 17:24:52 -07:00
|
|
|
fail fmt!("Expected failure, but got addr %?", addr);
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[test]
|
2012-06-27 15:28:03 -07:00
|
|
|
#[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-09-28 02:26:20 -07:00
|
|
|
result::Err(ref err_info) => {
|
2012-08-22 17:24:52 -07:00
|
|
|
log(debug, fmt!("got error as expected %?", err_info));
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
assert true;
|
|
|
|
}
|
2012-09-28 02:26:20 -07:00
|
|
|
result::Ok(ref addr) => {
|
2012-08-22 17:24:52 -07:00
|
|
|
fail fmt!("Expected failure, but got addr %?", addr);
|
std: adding net::ip::v6 utils and rudimentary tests, huzzah! (see caveats)
libuv's own ip vetting code appears to in a somewhat woeful state,
for both ipv4 and ipv6 (there are some notes in the tests for net_ip, as
well as stuff added in uv_ll). They are aware of this and welcome patches.
I have rudimentary code in place that can verify whether the provided str
ip was, in fact, validly parsed by libuv, making a few assumptions:
* for ipv4, we assume that the platform's INADDR_NONE val is 0xffffffff ,
I should write a helper to return this value from the platform's libc
headers instead of hard-coding it in rust.
* for ipv6, we assume that the library will always return '::' for
malformed inputs.. as is the case in 64bit ubuntu. I need to verify this
on other platforms.. but at least the debugging output is in place, so
if expectations don't line up, it'll be straightforward to address
2012-06-17 20:42:48 -07:00
|
|
|
}
|
|
|
|
}
|
2012-04-29 21:53:17 -07:00
|
|
|
}
|
2012-06-25 08:02:34 -07:00
|
|
|
#[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() {
|
2012-07-13 22:57:48 -07:00
|
|
|
let localhost_name = ~"localhost";
|
2013-01-19 23:38:17 -08:00
|
|
|
let iotask = &uv::global_loop::get();
|
2012-06-25 08:02:34 -07:00
|
|
|
let ga_result = get_addr(localhost_name, iotask);
|
2012-09-25 16:23:04 -07:00
|
|
|
if result::is_err(&ga_result) {
|
2012-07-13 22:57:48 -07:00
|
|
|
fail ~"got err result from net::ip::get_addr();"
|
2012-06-25 08:02:34 -07:00
|
|
|
}
|
|
|
|
// note really sure how to realiably test/assert
|
|
|
|
// this.. mostly just wanting to see it work, atm.
|
2012-09-18 22:35:42 -07:00
|
|
|
let results = result::unwrap(move 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)));
|
2012-09-18 21:41:13 -07:00
|
|
|
for vec::each(results) |r| {
|
2012-09-18 21:41:37 -07:00
|
|
|
let ipv_prefix = match *r {
|
2012-08-30 11:01:39 -07:00
|
|
|
Ipv4(_) => ~"IPv4",
|
|
|
|
Ipv6(_) => ~"IPv6"
|
2012-06-25 08:02:34 -07:00
|
|
|
};
|
2012-08-22 17:24:52 -07:00
|
|
|
log(debug, fmt!("test_get_addr: result %s: '%s'",
|
2012-09-18 21:41:37 -07:00
|
|
|
ipv_prefix, format_addr(r)));
|
2012-06-25 08:02:34 -07:00
|
|
|
}
|
2012-06-26 11:29:57 -07:00
|
|
|
// at least one result.. this is going to vary from system
|
|
|
|
// to system, based on stuff like the contents of /etc/hosts
|
2012-09-18 22:35:42 -07:00
|
|
|
assert !results.is_empty();
|
2012-04-29 21:53:17 -07:00
|
|
|
}
|
|
|
|
#[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() {
|
2012-07-13 22:57:48 -07:00
|
|
|
let localhost_name = ~"sjkl234m,./sdf";
|
2013-01-19 23:38:17 -08:00
|
|
|
let iotask = &uv::global_loop::get();
|
2012-06-26 11:29:57 -07:00
|
|
|
let ga_result = get_addr(localhost_name, iotask);
|
2012-09-25 16:23:04 -07:00
|
|
|
assert result::is_err(&ga_result);
|
2012-04-29 21:53:17 -07:00
|
|
|
}
|
2012-07-06 19:06:58 -07:00
|
|
|
}
|