2024-05-04 07:44:20 +02:00
|
|
|
use crate::database::cache;
|
2024-05-05 07:53:45 +02:00
|
|
|
use crate::util::http_client;
|
|
|
|
use isahc::ReadResponseExt;
|
2024-05-04 07:44:20 +02:00
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
|
|
|
|
#[derive(thiserror::Error, Debug)]
|
|
|
|
pub enum Error {
|
|
|
|
#[error("Cache error: {0}")]
|
|
|
|
CacheErr(#[from] cache::Error),
|
2024-05-05 07:53:45 +02:00
|
|
|
#[error("Isahc error: {0}")]
|
|
|
|
IsahcErr(#[from] isahc::Error),
|
|
|
|
#[error("HTTP client aquisition error: {0}")]
|
|
|
|
HttpClientErr(#[from] http_client::Error),
|
|
|
|
#[error("HTTP error: {0}")]
|
|
|
|
HttpErr(String),
|
|
|
|
#[error("Response parsing error: {0}")]
|
|
|
|
IoErr(#[from] std::io::Error),
|
2024-05-04 07:44:20 +02:00
|
|
|
#[error("Failed to deserialize JSON: {0}")]
|
|
|
|
JsonErr(#[from] serde_json::Error),
|
|
|
|
}
|
|
|
|
|
2024-05-04 15:49:11 +02:00
|
|
|
const UPSTREAM_PACKAGE_JSON_URL: &str =
|
2024-05-04 07:44:20 +02:00
|
|
|
"https://firefish.dev/firefish/firefish/-/raw/main/package.json";
|
|
|
|
|
|
|
|
async fn get_latest_version() -> Result<String, Error> {
|
|
|
|
#[derive(Debug, Deserialize, Serialize)]
|
|
|
|
struct Response {
|
|
|
|
version: String,
|
|
|
|
}
|
|
|
|
|
2024-05-05 07:53:45 +02:00
|
|
|
let mut response = http_client::client()?
|
|
|
|
.get(UPSTREAM_PACKAGE_JSON_URL)?;
|
|
|
|
|
|
|
|
if !response.status().is_success() {
|
|
|
|
tracing::info!("status: {}", response.status());
|
|
|
|
tracing::debug!("response body: {:#?}", response.body());
|
|
|
|
return Err(Error::HttpErr("Failed to fetch version from Firefish GitLab".to_string()));
|
|
|
|
}
|
|
|
|
|
|
|
|
let res_parsed: Response = serde_json::from_str(&response.text()?)?;
|
2024-05-04 07:44:20 +02:00
|
|
|
|
|
|
|
Ok(res_parsed.version)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[crate::export]
|
|
|
|
pub async fn latest_version() -> Result<String, Error> {
|
|
|
|
let version: Option<String> =
|
|
|
|
cache::get_one(cache::Category::FetchUrl, UPSTREAM_PACKAGE_JSON_URL)?;
|
|
|
|
|
|
|
|
if let Some(v) = version {
|
|
|
|
tracing::trace!("use cached value: {}", v);
|
|
|
|
Ok(v)
|
|
|
|
} else {
|
|
|
|
tracing::trace!("cache is expired, fetching the latest version");
|
|
|
|
let fetched_version = get_latest_version().await?;
|
|
|
|
tracing::trace!("fetched value: {}", fetched_version);
|
|
|
|
|
|
|
|
cache::set_one(
|
|
|
|
cache::Category::FetchUrl,
|
|
|
|
UPSTREAM_PACKAGE_JSON_URL,
|
|
|
|
&fetched_version,
|
|
|
|
3 * 60 * 60,
|
|
|
|
)?;
|
|
|
|
Ok(fetched_version)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod unit_test {
|
|
|
|
use super::{latest_version, UPSTREAM_PACKAGE_JSON_URL};
|
|
|
|
use crate::database::cache;
|
|
|
|
|
|
|
|
fn validate_version(version: String) {
|
2024-05-04 09:17:33 +02:00
|
|
|
// version: YYYYMMDD or YYYYMMDD-X
|
|
|
|
assert!(version.len() >= 8);
|
|
|
|
assert!(version[..8].chars().all(|c| c.is_ascii_digit()));
|
2024-05-04 07:44:20 +02:00
|
|
|
|
|
|
|
// YYYY
|
|
|
|
assert!(&version[..4] >= "2024");
|
|
|
|
|
|
|
|
// MM
|
|
|
|
assert!(&version[4..6] >= "01");
|
|
|
|
assert!(&version[4..6] <= "12");
|
|
|
|
|
|
|
|
// DD
|
2024-05-04 09:17:33 +02:00
|
|
|
assert!(&version[6..8] >= "01");
|
|
|
|
assert!(&version[6..8] <= "31");
|
2024-05-04 15:59:49 +02:00
|
|
|
|
|
|
|
// -X
|
|
|
|
if version.len() > 8 {
|
|
|
|
assert!(version.chars().nth(8).unwrap() == '-');
|
|
|
|
assert!(version[9..].chars().all(|c| c.is_ascii_digit()));
|
|
|
|
}
|
2024-05-04 07:44:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test]
|
|
|
|
async fn check_version() {
|
2024-05-04 15:59:49 +02:00
|
|
|
// delete caches in case you run this test multiple times
|
2024-05-04 14:24:20 +02:00
|
|
|
cache::delete_one(cache::Category::FetchUrl, UPSTREAM_PACKAGE_JSON_URL).unwrap();
|
2024-05-04 07:44:20 +02:00
|
|
|
|
|
|
|
// fetch from firefish.dev
|
|
|
|
validate_version(latest_version().await.unwrap());
|
|
|
|
|
|
|
|
// use cache
|
|
|
|
validate_version(latest_version().await.unwrap());
|
|
|
|
}
|
|
|
|
}
|