mirror of
https://github.com/achanda/ipnetwork.git
synced 2025-06-16 08:48:51 +00:00
168 lines
3.9 KiB
Rust
168 lines
3.9 KiB
Rust
#![cfg_attr(feature = "dev", allow(unstable_features))]
|
|
#![cfg_attr(feature = "dev", feature(plugin))]
|
|
#![cfg_attr(feature = "dev", plugin(clippy))]
|
|
#![crate_type = "lib"]
|
|
#[allow(dead_code)]
|
|
|
|
use std::fmt;
|
|
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr};
|
|
|
|
// A network
|
|
#[derive(Debug)]
|
|
pub enum IpNetwork {
|
|
V4(Ipv4Network),
|
|
V6(Ipv6Network),
|
|
}
|
|
|
|
pub struct Ipv4Network {
|
|
addr: Ipv4Addr,
|
|
prefix: u8,
|
|
}
|
|
|
|
pub struct Ipv6Network {
|
|
addr: Ipv6Addr,
|
|
prefix: u8,
|
|
}
|
|
|
|
impl Ipv4Network {
|
|
pub fn new(addr: Ipv4Addr, prefix: u8) -> Ipv4Network {
|
|
Ipv4Network {
|
|
addr: addr,
|
|
prefix: prefix,
|
|
}
|
|
}
|
|
|
|
pub fn ip(&self) -> Ipv4Addr {
|
|
self.addr
|
|
}
|
|
|
|
pub fn prefix(&self) -> u8 {
|
|
self.prefix
|
|
}
|
|
|
|
pub fn mask(&self) -> (Ipv4Addr, u32) {
|
|
let prefix = self.prefix;
|
|
let mask = !(0xffffffff >> prefix);
|
|
(Ipv4Addr::from(mask), mask)
|
|
}
|
|
|
|
pub fn network(&self) -> (Ipv4Addr, u32) {
|
|
let (_, mask) = self.mask();
|
|
let ip = u32::from(self.addr) & mask;
|
|
(Ipv4Addr::from(ip), ip)
|
|
}
|
|
|
|
pub fn broadcast(&self) -> (Ipv4Addr, u32) {
|
|
let (_, network) = self.network();
|
|
let (_, mask) = self.mask();
|
|
let broadcast = network | !mask;
|
|
(Ipv4Addr::from(broadcast), broadcast)
|
|
}
|
|
|
|
pub fn contains(&self, ip: Ipv4Addr) -> bool {
|
|
let (_, net) = self.network();
|
|
(u32::from(ip) & net) == net
|
|
}
|
|
}
|
|
|
|
impl Ipv6Network {
|
|
pub fn new(addr: Ipv6Addr, prefix: u8) -> Ipv6Network {
|
|
Ipv6Network {
|
|
addr: addr,
|
|
prefix: prefix,
|
|
}
|
|
}
|
|
|
|
pub fn ip(&self) -> Ipv6Addr {
|
|
self.addr
|
|
}
|
|
|
|
pub fn prefix(&self) -> u8 {
|
|
self.prefix
|
|
}
|
|
}
|
|
|
|
impl IpNetwork {
|
|
pub fn new(ip: IpAddr, prefix: u8) -> IpNetwork {
|
|
match ip {
|
|
IpAddr::V4(a) => IpNetwork::V4(Ipv4Network::new(a, prefix)),
|
|
IpAddr::V6(a) => IpNetwork::V6(Ipv6Network::new(a, prefix)),
|
|
}
|
|
}
|
|
|
|
pub fn ip(&self) -> IpAddr {
|
|
match *self {
|
|
IpNetwork::V4(ref a) => IpAddr::V4(a.ip()),
|
|
IpNetwork::V6(ref a) => IpAddr::V6(a.ip()),
|
|
}
|
|
}
|
|
|
|
pub fn prefix(&self) -> u8 {
|
|
match *self {
|
|
IpNetwork::V4(ref a) => a.prefix(),
|
|
IpNetwork::V6(ref a) => a.prefix(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for Ipv4Network {
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(fmt, "{}/{}", self.ip(), self.prefix())
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for Ipv6Network {
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(fmt, "{}/{}", self.ip(), self.prefix())
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod test {
|
|
use std::net::{Ipv4Addr, Ipv6Addr};
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn create_v4() {
|
|
let cidr = Ipv4Network::new(Ipv4Addr::new(77, 88, 21, 11), 24);
|
|
assert_eq!(cidr.prefix(), 24);
|
|
}
|
|
|
|
#[test]
|
|
fn create_v6() {
|
|
let cidr = Ipv6Network::new(Ipv6Addr::new(0, 0, 0, 0, 0, 0, 0, 1), 24);
|
|
assert_eq!(cidr.prefix(), 24);
|
|
}
|
|
|
|
#[test]
|
|
fn mask_v4() {
|
|
let cidr = Ipv4Network::new(Ipv4Addr::new(74, 125, 227, 0), 29);
|
|
let (ip, int) = cidr.mask();
|
|
assert_eq!(ip, Ipv4Addr::new(255, 255, 255, 248));
|
|
assert_eq!(int, 4294967288);
|
|
}
|
|
|
|
#[test]
|
|
fn network_v4() {
|
|
let cidr = Ipv4Network::new(Ipv4Addr::new(10, 10, 1, 97), 23);
|
|
let (ip, int) = cidr.network();
|
|
assert_eq!(ip, Ipv4Addr::new(10, 10, 0, 0));
|
|
assert_eq!(int, 168427520);
|
|
}
|
|
|
|
#[test]
|
|
fn broadcast_v4() {
|
|
let cidr = Ipv4Network::new(Ipv4Addr::new(10, 10, 1, 97), 23);
|
|
let (ip, int) = cidr.broadcast();
|
|
assert_eq!(ip, Ipv4Addr::new(10, 10, 1, 255));
|
|
assert_eq!(int, 168428031);
|
|
}
|
|
|
|
#[test]
|
|
fn contains_v4() {
|
|
let cidr = Ipv4Network::new(Ipv4Addr::new(74, 125, 227, 0), 25);
|
|
let ip = Ipv4Addr::new(74, 125, 227, 4);
|
|
assert!(cidr.contains(ip));
|
|
}
|
|
}
|