use std::{
collections::HashMap,
net::{Ipv4Addr, Ipv6Addr},
};
use prost::Message;
pub fn decode_geosite(buf: &[u8]) -> Result<GeoSiteList, prost::DecodeError> {
GeoSiteList::decode(buf)
}
pub fn encode_geosite(sg: GeoSiteList) -> Vec<u8> {
sg.encode_to_vec()
}
pub fn decode_geoip(buf: &[u8]) -> Result<GeoIpList, prost::DecodeError> {
GeoIpList::decode(buf)
}
pub fn encode_geoip(sg: GeoIpList) -> Vec<u8> {
sg.encode_to_vec()
}
fn vec_to_u32_be(bytes: &[u8]) -> u32 {
assert!(bytes.len() == 4, "Input Vec<u8> must have exactly 4 bytes");
let mut array = [0u8; 4];
array.copy_from_slice(bytes);
u32::from_be_bytes(array)
}
fn vec_to_u128_be(bytes: &[u8]) -> u128 {
assert!(
bytes.len() == 16,
"Input Vec<u8> must have exactly 16 bytes"
);
let mut array = [0u8; 16];
array.copy_from_slice(bytes);
u128::from_be_bytes(array)
}
pub fn geoip_to_hashmap(
geoip_list: &GeoIpList,
country_target_map: HashMap<String, String>,
) -> HashMap<String, Vec<Vec<String>>> {
let mut map: HashMap<String, Vec<Vec<String>>> = HashMap::new();
for ipg in &geoip_list.entry {
for cidr in &ipg.cidr {
let ipn = &cidr.ip;
if ipn.len() == 4 {
let ia = Ipv4Addr::from(vec_to_u32_be(ipn));
let is = ia.to_string();
let s = format!("{is}/{}", cidr.prefix);
let v = vec![
s,
country_target_map
.get(&ipg.country_code)
.unwrap()
.to_string(),
];
map.entry("IP-CIDR".to_string()).or_default().push(v);
} else if ipn.len() == 16 {
let ia = Ipv6Addr::from(vec_to_u128_be(ipn));
let is = ia.to_string();
let s = format!("{is}/{}", cidr.prefix);
let v = vec![
s,
country_target_map
.get(&ipg.country_code)
.unwrap()
.to_string(),
];
map.entry("IP-CIDR6".to_string()).or_default().push(v);
}
}
}
map
}
pub fn geosite_to_hashmap(
site_group_list: &GeoSiteList,
group_target_map: HashMap<String, String>,
) -> HashMap<String, Vec<Vec<String>>> {
let mut map: HashMap<String, Vec<Vec<String>>> = HashMap::new();
for group in &site_group_list.entry {
for domain in &group.domain {
let key = match domain.r#type {
0 => "DOMAIN-KEYWORD", 1 => "DOMAIN-SUFFIX", 2 => "DOMAIN", 3 => "DOMAIN-REGEX", _ => continue, };
let v = vec![
domain.value.clone(),
group_target_map
.get(&group.country_code)
.unwrap()
.to_string(),
];
map.entry(key.to_string()).or_default().push(v);
}
}
map
}
#[derive(Clone, PartialEq, ::prost::Message)]
pub struct Domain {
#[prost(enumeration = "domain::Type", tag = "1")]
pub r#type: i32,
#[prost(string, tag = "2")]
pub value: ::prost::alloc::string::String,
#[prost(message, repeated, tag = "3")]
pub attribute: ::prost::alloc::vec::Vec<domain::Attribute>,
}
pub mod domain {
#[derive(Clone, PartialEq, ::prost::Message)]
pub struct Attribute {
#[prost(string, tag = "1")]
pub key: ::prost::alloc::string::String,
#[prost(oneof = "attribute::TypedValue", tags = "2, 3")]
pub typed_value: ::core::option::Option<attribute::TypedValue>,
}
pub mod attribute {
#[derive(Clone, Copy, PartialEq, ::prost::Oneof)]
pub enum TypedValue {
#[prost(bool, tag = "2")]
BoolValue(bool),
#[prost(int64, tag = "3")]
IntValue(i64),
}
}
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, PartialOrd, Ord, ::prost::Enumeration)]
#[repr(i32)]
pub enum Type {
Plain = 0,
Regex = 1,
Domain = 2,
Full = 3,
}
impl Type {
pub fn as_str_name(&self) -> &'static str {
match self {
Self::Plain => "Plain",
Self::Regex => "Regex",
Self::Domain => "Domain",
Self::Full => "Full",
}
}
pub fn from_str_name(value: &str) -> ::core::option::Option<Self> {
match value {
"Plain" => Some(Self::Plain),
"Regex" => Some(Self::Regex),
"Domain" => Some(Self::Domain),
"Full" => Some(Self::Full),
_ => None,
}
}
}
}
#[derive(Clone, PartialEq, ::prost::Message)]
pub struct GeoSite {
#[prost(string, tag = "1")]
pub country_code: ::prost::alloc::string::String,
#[prost(message, repeated, tag = "2")]
pub domain: ::prost::alloc::vec::Vec<Domain>,
}
#[derive(Clone, PartialEq, ::prost::Message)]
pub struct GeoSiteList {
#[prost(message, repeated, tag = "1")]
pub entry: ::prost::alloc::vec::Vec<GeoSite>,
}
#[derive(Clone, PartialEq, ::prost::Message)]
pub struct Cidr {
#[prost(bytes = "vec", tag = "1")]
pub ip: ::prost::alloc::vec::Vec<u8>,
#[prost(uint32, tag = "2")]
pub prefix: u32,
}
#[derive(Clone, PartialEq, ::prost::Message)]
pub struct GeoIp {
#[prost(string, tag = "1")]
pub country_code: ::prost::alloc::string::String,
#[prost(message, repeated, tag = "2")]
pub cidr: ::prost::alloc::vec::Vec<Cidr>,
#[prost(bool, tag = "3")]
pub reverse_match: bool,
}
#[derive(Clone, PartialEq, ::prost::Message)]
pub struct GeoIpList {
#[prost(message, repeated, tag = "1")]
pub entry: ::prost::alloc::vec::Vec<GeoIp>,
}