use crate::{download_file, Error};
use crate::minecraft::{Argument, ArgumentType, Library, VersionInfo, VersionType};
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use std::collections::HashMap;
pub const CURRENT_FABRIC_FORMAT_VERSION: usize = 0;
pub const CURRENT_FORGE_FORMAT_VERSION: usize = 0;
#[derive(Serialize, Deserialize, Debug)]
pub struct SidedDataEntry {
pub client: String,
pub server: String,
}
#[derive(Serialize, Deserialize, Debug)]
#[serde(rename_all = "camelCase")]
pub struct PartialVersionInfo {
pub id: String,
pub inherits_from: String,
pub release_time: DateTime<Utc>,
pub time: DateTime<Utc>,
#[serde(skip_serializing_if = "Option::is_none")]
pub main_class: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub minecraft_arguments: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub arguments: Option<HashMap<ArgumentType, Vec<Argument>>>,
pub libraries: Vec<Library>,
#[serde(rename = "type")]
pub type_: VersionType,
#[serde(skip_serializing_if = "Option::is_none")]
pub data: Option<HashMap<String, SidedDataEntry>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub processors: Option<Vec<Processor>>,
}
#[derive(Serialize, Deserialize, Debug)]
pub struct Processor {
pub jar: String,
pub classpath: Vec<String>,
pub args: Vec<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub outputs: Option<HashMap<String, String>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub sides: Option<Vec<String>>,
}
pub async fn fetch_partial_version(url: &str) -> Result<PartialVersionInfo, Error> {
Ok(serde_json::from_slice(&download_file(url, None).await?)?)
}
pub fn merge_partial_version(partial: PartialVersionInfo, merge: VersionInfo) -> VersionInfo {
VersionInfo {
arguments: if let Some(partial_args) = partial.arguments {
if let Some(merge_args) = merge.arguments {
let mut new_map = HashMap::new();
fn add_keys(
new_map: &mut HashMap<ArgumentType, Vec<Argument>>,
args: HashMap<ArgumentType, Vec<Argument>>,
) {
for (type_, arguments) in args {
for arg in arguments {
if let Some(vec) = new_map.get_mut(&type_) {
vec.push(arg);
} else {
new_map.insert(type_, vec![arg]);
}
}
}
}
add_keys(&mut new_map, merge_args);
add_keys(&mut new_map, partial_args);
Some(new_map)
} else {
Some(partial_args)
}
} else {
merge.arguments
},
asset_index: merge.asset_index,
assets: merge.assets,
downloads: merge.downloads,
id: partial.id,
libraries: partial
.libraries
.into_iter()
.chain(merge.libraries)
.collect::<Vec<_>>(),
main_class: if let Some(main_class) = partial.main_class {
main_class
} else {
merge.main_class
},
minecraft_arguments: partial.minecraft_arguments,
minimum_launcher_version: merge.minimum_launcher_version,
release_time: partial.release_time,
time: partial.time,
type_: partial.type_,
data: partial.data,
processors: partial.processors,
}
}
#[derive(Serialize, Deserialize, Debug, Clone)]
#[serde(rename_all = "camelCase")]
pub struct Manifest {
pub game_versions: Vec<Version>,
}
#[derive(Serialize, Deserialize, Debug, Eq, PartialEq, Hash, Clone)]
#[serde(rename_all = "camelCase")]
pub enum LoaderType {
Latest,
Stable,
}
#[derive(Serialize, Deserialize, Debug, Clone)]
pub struct Version {
pub id: String,
pub loaders: HashMap<LoaderType, LoaderVersion>,
}
#[derive(Serialize, Deserialize, Debug, Clone)]
pub struct LoaderVersion {
pub id: String,
pub url: String,
}
pub async fn fetch_manifest(url: &str) -> Result<Manifest, Error> {
Ok(serde_json::from_slice(&download_file(url, None).await?)?)
}