You've already forked AstralRinth
forked from didirus/AstralRinth
* Update Rust version * Update async-compression 0.4.25 -> 0.4.27 * Update async-tungstenite 0.29.1 -> 0.30.0 * Update bytemuck 1.23.0 -> 1.23.1 * Update clap 4.5.40 -> 4.5.43 * Update deadpool-redis 0.21.1 -> 0.22.0 and redis 0.31.0 -> 0.32.4 * Update enumset 1.1.6 -> 1.1.7 * Update hyper-util 0.1.14 -> 0.1.16 * Update indexmap 2.9.0 -> 2.10.0 * Update indicatif 0.17.11 -> 0.18.0 * Update jemalloc_pprof 0.7.0 -> 0.8.1 * Update lettre 0.11.17 -> 0.11.18 * Update meilisearch-sdk 0.28.0 -> 0.29.1 * Update notify 8.0.0 -> 8.2.0 and notify-debouncer-mini 0.6.0 -> 0.7.0 * Update quick-xml 0.37.5 -> 0.38.1 * Fix theseus lint * Update reqwest 0.12.20 -> 0.12.22 * Cargo fmt in theseus * Update rgb 0.8.50 -> 0.8.52 * Update sentry 0.41.0 -> 0.42.0 and sentry-actix 0.41.0 -> 0.42.0 * Update serde_json 1.0.140 -> 1.0.142 * Update serde_with 3.13.0 -> 3.14.0 * Update spdx 0.10.8 -> 0.10.9 * Update sysinfo 0.35.2 -> 0.36.1 * Update tauri suite * Fix build by updating mappings * Update tokio 1.45.1 -> 1.47.1 and tokio-util 0.7.15 -> 0.7.16 * Update tracing-actix-web 0.7.18 -> 0.7.19 * Update zip 4.2.0 -> 4.3.0 * Misc Cargo.lock updates * Update Dockerfiles
88 lines
2.5 KiB
Rust
88 lines
2.5 KiB
Rust
use flate2::read::GzDecoder;
|
|
use maxminddb::geoip2::Country;
|
|
use std::io::{Cursor, Read};
|
|
use std::net::Ipv6Addr;
|
|
use tar::Archive;
|
|
use tokio::sync::RwLock;
|
|
use tracing::warn;
|
|
|
|
pub struct MaxMindIndexer {
|
|
pub reader: RwLock<Option<maxminddb::Reader<Vec<u8>>>>,
|
|
}
|
|
|
|
impl MaxMindIndexer {
|
|
pub async fn new() -> Result<Self, reqwest::Error> {
|
|
let reader = MaxMindIndexer::inner_index(false).await.ok().flatten();
|
|
|
|
Ok(MaxMindIndexer {
|
|
reader: RwLock::new(reader),
|
|
})
|
|
}
|
|
|
|
pub async fn index(&self) -> Result<(), reqwest::Error> {
|
|
let reader = MaxMindIndexer::inner_index(false).await?;
|
|
|
|
if let Some(reader) = reader {
|
|
let mut reader_new = self.reader.write().await;
|
|
*reader_new = Some(reader);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn inner_index(
|
|
should_panic: bool,
|
|
) -> Result<Option<maxminddb::Reader<Vec<u8>>>, reqwest::Error> {
|
|
let response = reqwest::get(
|
|
format!(
|
|
"https://download.maxmind.com/app/geoip_download?edition_id=GeoLite2-Country&license_key={}&suffix=tar.gz",
|
|
dotenvy::var("MAXMIND_LICENSE_KEY").unwrap()
|
|
)
|
|
).await?.bytes().await.unwrap().to_vec();
|
|
|
|
let tarfile = GzDecoder::new(Cursor::new(response));
|
|
let mut archive = Archive::new(tarfile);
|
|
|
|
if let Ok(entries) = archive.entries() {
|
|
for mut file in entries.flatten() {
|
|
if let Ok(path) = file.header().path()
|
|
&& path.extension().and_then(|x| x.to_str()) == Some("mmdb")
|
|
{
|
|
let mut buf = Vec::new();
|
|
file.read_to_end(&mut buf).unwrap();
|
|
|
|
let reader = maxminddb::Reader::from_source(buf).unwrap();
|
|
|
|
return Ok(Some(reader));
|
|
}
|
|
}
|
|
}
|
|
|
|
if should_panic {
|
|
panic!(
|
|
"Unable to download maxmind database- did you get a license key?"
|
|
)
|
|
} else {
|
|
warn!("Unable to download maxmind database.");
|
|
|
|
Ok(None)
|
|
}
|
|
}
|
|
|
|
pub async fn query(&self, ip: Ipv6Addr) -> Option<String> {
|
|
let maxmind = self.reader.read().await;
|
|
|
|
if let Some(ref maxmind) = *maxmind {
|
|
maxmind
|
|
.lookup::<Country>(ip.into())
|
|
.ok()
|
|
.flatten()
|
|
.and_then(|x| {
|
|
x.country.and_then(|x| x.iso_code.map(|x| x.to_string()))
|
|
})
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
}
|