1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
use cassandra::error::CassError;
use cassandra::util::Protected;
use cassandra_sys::CASS_OK;
use cassandra_sys::CassInet as _Inet;
use cassandra_sys::cass_inet_from_string;
use cassandra_sys::cass_inet_init_v4;
use cassandra_sys::cass_inet_init_v6;
use cassandra_sys::cass_inet_string;
use errors::*;
use std::default::Default;
use std::ffi::CStr;
use std::ffi::CString;
use std::fmt;
use std::fmt::{Debug, Formatter};
use std::mem;
use std::net::{Ipv4Addr, Ipv6Addr};
use std::net::SocketAddr;
use std::str::FromStr;
use std::string::ToString;
#[repr(C)]
pub struct Inet(_Inet);
impl Debug for Inet {
fn fmt(&self, f: &mut Formatter) -> fmt::Result { write!(f, "can't format an inet") }
}
impl Protected<_Inet> for Inet {
fn inner(&self) -> _Inet { self.0 }
fn build(inner: _Inet) -> Self { Inet(inner) }
}
impl Default for Inet {
fn default() -> Inet { unsafe { ::std::mem::zeroed() } }
}
pub trait AsInet {
fn as_cass_inet(&self) -> Inet;
}
impl AsInet for SocketAddr {
fn as_cass_inet(&self) -> Inet {
match *self {
SocketAddr::V4(ipv4_addr) => unsafe { Inet(cass_inet_init_v4(ipv4_addr.ip().octets().as_ptr())) },
SocketAddr::V6(ipv6_addr) => {
unsafe {
let seg = ipv6_addr.ip().segments();
Inet(cass_inet_init_v6(seg.as_ptr() as *const u8))
}
}
}
}
}
impl FromStr for Inet {
type Err = Error;
fn from_str(s: &str) -> Result<Self> {
unsafe {
let mut inet = mem::zeroed();
let str = CString::new(s).expect("must be utf8");
match cass_inet_from_string(str.as_ptr(), &mut inet) {
CASS_OK => Ok(Inet(inet)),
err => err.to_result(Inet(inet)).chain_err(|| ""),
}
}
}
}
impl ToString for Inet {
fn to_string(&self) -> String {
unsafe {
let mut inet_str = mem::zeroed();
cass_inet_string(self.0, &mut inet_str);
CStr::from_ptr(&inet_str).to_string_lossy().into_owned()
}
}
}
pub trait FromInet {
fn from_cass_inet(inet: Inet) -> Self;
}
impl FromInet for Ipv4Addr {
fn from_cass_inet(inet: Inet) -> Self {
let raw_addr: [u8; 16] = inet.0.address;
match inet.0.address_length {
4 => Ipv4Addr::new(raw_addr[0], raw_addr[1], raw_addr[2], raw_addr[3]),
16 => panic!(),
unsupported => panic!("impossible inet type: {:?}", unsupported),
}
}
}
impl FromInet for Ipv6Addr {
fn from_cass_inet(inet: Inet) -> Self {
let raw_addr: [u8; 16] = inet.0.address;
match inet.0.address_length {
4 => panic!(),
16 => {
Ipv6Addr::new((raw_addr[1] as u16) << (8 + raw_addr[0] as u16),
(raw_addr[3] as u16) << (8 + raw_addr[2] as u16),
(raw_addr[5] as u16) << (8 + raw_addr[4] as u16),
(raw_addr[7] as u16) << (8 + raw_addr[6] as u16),
(raw_addr[9] as u16) << (8 + raw_addr[8] as u16),
(raw_addr[11] as u16) << (8 + raw_addr[10] as u16),
(raw_addr[13] as u16) << (8 + raw_addr[12] as u16),
(raw_addr[15] as u16) << (8 + raw_addr[14] as u16))
}
unsupported => panic!("impossible inet type: {}", unsupported),
}
}
}
impl Inet {
pub fn cass_inet_init_v4(address: Ipv4Addr) -> Inet {
unsafe { Inet(cass_inet_init_v4(address.octets().as_ptr())) }
}
pub fn cass_inet_init_v6(address: Ipv6Addr) -> Inet {
unsafe { Inet(cass_inet_init_v6(address.segments().as_ptr() as *const u8)) }
}
}