You've already forked AstralRinth
forked from didirus/AstralRinth
138 lines
3.8 KiB
Rust
138 lines
3.8 KiB
Rust
//! Theseus metadata
|
|
use crate::data::DirectoryInfo;
|
|
use crate::util::fetch::{read_json, write, IoSemaphore};
|
|
use crate::State;
|
|
use daedalus::{
|
|
minecraft::{fetch_version_manifest, VersionManifest as MinecraftManifest},
|
|
modded::{
|
|
fetch_manifest as fetch_loader_manifest, Manifest as LoaderManifest,
|
|
},
|
|
};
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
const METADATA_URL: &str = "https://meta.modrinth.com";
|
|
|
|
#[derive(Serialize, Deserialize, Debug, Clone)]
|
|
pub struct Metadata {
|
|
pub minecraft: MinecraftManifest,
|
|
pub forge: LoaderManifest,
|
|
pub fabric: LoaderManifest,
|
|
pub quilt: LoaderManifest,
|
|
}
|
|
|
|
impl Metadata {
|
|
fn get_manifest(name: &str) -> String {
|
|
format!("{METADATA_URL}/{name}/v0/manifest.json")
|
|
}
|
|
|
|
pub async fn fetch() -> crate::Result<Self> {
|
|
let (minecraft, forge, fabric, quilt) = tokio::try_join! {
|
|
async {
|
|
let url = Self::get_manifest("minecraft");
|
|
fetch_version_manifest(Some(&url)).await
|
|
},
|
|
async {
|
|
let url = Self::get_manifest("forge");
|
|
fetch_loader_manifest(&url).await
|
|
},
|
|
async {
|
|
let url = Self::get_manifest("fabric");
|
|
fetch_loader_manifest(&url).await
|
|
},
|
|
async {
|
|
let url = Self::get_manifest("quilt");
|
|
fetch_loader_manifest(&url).await
|
|
}
|
|
}?;
|
|
|
|
Ok(Self {
|
|
minecraft,
|
|
forge,
|
|
fabric,
|
|
quilt,
|
|
})
|
|
}
|
|
|
|
// Attempt to fetch metadata and store in sled DB
|
|
#[tracing::instrument(skip(io_semaphore))]
|
|
#[theseus_macros::debug_pin]
|
|
pub async fn init(
|
|
dirs: &DirectoryInfo,
|
|
io_semaphore: &IoSemaphore,
|
|
) -> crate::Result<Self> {
|
|
let mut metadata = None;
|
|
let metadata_path = dirs.caches_meta_dir().await.join("metadata.json");
|
|
|
|
if let Ok(metadata_json) =
|
|
read_json::<Metadata>(&metadata_path, io_semaphore).await
|
|
{
|
|
metadata = Some(metadata_json);
|
|
} else {
|
|
let res = async {
|
|
let metadata_fetch = Self::fetch().await?;
|
|
|
|
write(
|
|
&metadata_path,
|
|
&serde_json::to_vec(&metadata_fetch).unwrap_or_default(),
|
|
io_semaphore,
|
|
)
|
|
.await?;
|
|
|
|
metadata = Some(metadata_fetch);
|
|
Ok::<(), crate::Error>(())
|
|
}
|
|
.await;
|
|
|
|
match res {
|
|
Ok(()) => {}
|
|
Err(err) => {
|
|
tracing::warn!("Unable to fetch launcher metadata: {err}")
|
|
}
|
|
}
|
|
}
|
|
|
|
if let Some(meta) = metadata {
|
|
Ok(meta)
|
|
} else {
|
|
Err(
|
|
crate::ErrorKind::NoValueFor(String::from("launcher metadata"))
|
|
.as_error(),
|
|
)
|
|
}
|
|
}
|
|
|
|
pub async fn update() {
|
|
let res = async {
|
|
let metadata_fetch = Metadata::fetch().await?;
|
|
let state = State::get().await?;
|
|
|
|
let metadata_path = state
|
|
.directories
|
|
.caches_meta_dir()
|
|
.await
|
|
.join("metadata.json");
|
|
|
|
write(
|
|
&metadata_path,
|
|
&serde_json::to_vec(&metadata_fetch)?,
|
|
&state.io_semaphore,
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let mut old_metadata = state.metadata.write().await;
|
|
*old_metadata = metadata_fetch;
|
|
|
|
Ok::<(), crate::Error>(())
|
|
}
|
|
.await;
|
|
|
|
match res {
|
|
Ok(()) => {}
|
|
Err(err) => {
|
|
tracing::warn!("Unable to update launcher metadata: {err}")
|
|
}
|
|
};
|
|
}
|
|
}
|