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
#[macro_use]
extern crate lazy_static;
use regex::Regex;
use std::net::Ipv4Addr;
#[derive(Debug)]
struct IpMateData {
ip: u32,
sub_mark: u8,
}
#[derive(Debug)]
pub struct IpPart {
ip: IpMateData,
}
lazy_static! {
static ref RE: Regex =
Regex::new(r"^(\d{1,3})\.(\d{1,3})\.(\d{1,3})\.(\d{1,3})/(\d{1,3})$").unwrap();
}
fn u32_to_ipv4(ip: u32) -> Ipv4Addr {
let mut u: [u8; 4] = [0; 4];
for i in 0..4 {
u[i] = ((ip >> (i * 8)) & 0xff) as u8;
}
Ipv4Addr::new(u[3], u[2], u[1], u[0])
}
impl IpPart {
pub fn new(ip: &str) -> Result<Self, String> {
if RE.is_match(ip) {
for cap in RE.captures_iter(ip) {
let nums: Vec<u8> = [&cap[1], &cap[2], &cap[3], &cap[4], &cap[5]]
.iter()
.map(|n| n.parse().unwrap())
.collect();
if nums[4] > 32 {
return Err(String::from("子网掩码不能大于32"));
}
let mut ip: u32 = (nums[0] as u32) << 24;
ip += (nums[1] as u32) << 16;
ip += (nums[2] as u32) << 8;
ip += nums[3] as u32;
ip = ip & !((1 << (32 - nums[4])) - 1);
return Ok(IpPart {
ip: IpMateData {
ip: ip,
sub_mark: nums[4],
},
});
}
}
Err(String::from("ip格式错误"))
}
pub fn size(&self) -> u32 {
1 << (32 - self.ip.sub_mark)
}
pub fn list(&self) -> Vec<Ipv4Addr> {
let mut list: Vec<Ipv4Addr> = Vec::new();
for i in 0..self.size() {
list.push(u32_to_ipv4(self.ip.ip + i));
}
list
}
}
#[test]
fn test() {
let ip = IpPart::new("192.168.1.1/24").unwrap();
println!("{:?} size {}", ip.list(), ip.size());
}