Initial commit

Includes IPv4 and IPv6 support, and a CLI for generating IpRanges from ip2location DB1 CSV files.
This commit is contained in:
2025-03-11 21:16:25 -06:00
commit ebb7bbdf47
17 changed files with 789 additions and 0 deletions
+26
View File
@@ -0,0 +1,26 @@
[package]
name = "ipfilter-cli"
version = "0.0.0"
authors.workspace = true
edition.workspace = true
license-file.workspace = true
rust-version.workspace = true
[lints]
workspace = true
[dependencies]
ipfilter = { workspace = true, features = ["std"] }
anyhow = { workspace = true }
clap = { workspace = true, features = [
"derive",
"color",
"help",
"suggestions",
"usage",
] }
csv = { workspace = true }
ipnet = { workspace = true }
itertools = { workspace = true }
+38
View File
@@ -0,0 +1,38 @@
use std::path::PathBuf;
use clap::{Parser, Subcommand};
#[derive(Parser)]
#[command(version, about, long_about = None)]
pub struct Cli {
#[arg(short, long, value_name = "FILE")]
pub input: PathBuf,
#[command(subcommand)]
pub command: Commands,
}
#[derive(Subcommand)]
pub enum Commands {
/// Print a list of all countries and their codes
List,
/// Merge country IP blocks
Merge {
/// Comma separated list of country codes
#[arg(short, long)]
countries: String,
/// IPv6
#[arg(short, default_value = "false")]
_6: bool,
#[arg(short, long, value_name = "FILE")]
output: Option<PathBuf>,
},
/// Load a
Load {
/// IPv6
#[arg(short, default_value = "false")]
_6: bool,
},
}
+109
View File
@@ -0,0 +1,109 @@
use core::net::{Ipv4Addr, Ipv6Addr};
use std::collections::HashSet;
use std::fs;
use ipfilter::{v4, v6};
use anyhow::Result;
use clap::Parser;
use ipnet::{Ipv4Subnets, Ipv6Subnets};
mod cli;
use cli::{Cli, Commands};
use itertools::Itertools;
fn main() -> Result<()> {
let cli = Cli::parse();
let input = fs::File::open(cli.input)?;
match cli.command {
Commands::List => {
let mut reader = csv::ReaderBuilder::new()
.has_headers(false)
.from_reader(input);
let mut countries = HashSet::<(String, String)>::new();
for result in reader.records() {
let record = result?;
countries.insert((record[2].to_owned(), record[3].to_owned()));
}
let mut countries: Vec<_> = countries.drain().collect();
countries.sort_by(|lhs, rhs| lhs.0.cmp(&rhs.0));
for (code, country) in countries {
println!("{code} - {country}");
}
}
Commands::Merge {
countries,
output,
_6,
} => {
let mut reader = csv::ReaderBuilder::new()
.has_headers(false)
.from_reader(input);
let countries: HashSet<String> = countries.split(',').map(ToOwned::to_owned).collect();
let records = reader
.records()
.filter_ok(|r| countries.contains(&r[2]))
.map(|r| r.map_err(anyhow::Error::from));
macro_rules! merge_ip {
($v:ident, $net:ty, $addr:ty) => {
let r = records
.map(|r| {
r.and_then(|record| {
Ok(<$net>::new(
<$addr>::from_bits(record[0].parse()?),
<$addr>::from_bits(record[1].parse()?),
0,
))
})
})
.flatten_ok();
let range = $v::from_fallible_iter(r)?;
if let Some(o) = output {
let mut writer = fs::File::create(o)?;
$v::write_to(range, &mut writer)?;
} else {
for subnet in &range {
println!("{subnet}");
}
};
};
}
if _6 {
merge_ip!(v6, Ipv6Subnets, Ipv6Addr);
} else {
merge_ip!(v4, Ipv4Subnets, Ipv4Addr);
}
}
Commands::Load { _6 } => {
macro_rules! load_ip {
($v:ident, $net:ty, $addr:ty) => {
let range = $v::read_from(&mut &input)?;
for subnet in &range {
println!("{subnet}");
}
};
}
if _6 {
load_ip!(v6, Ipv6Subnets, Ipv6Addr);
} else {
load_ip!(v4, Ipv4Subnets, Ipv4Addr);
}
}
}
Ok(())
}