summaryrefslogblamecommitdiff
path: root/src/commands/update.rs
blob: e383eaef87e80056f08dc1974461c9dddc4a0061 (plain) (tree)
1
2
3
4
5
6
7
                                                   
 



                            
                                                                                                                                                                                                                                       




                                                                            
                                                                              
 
                                                                        
                                                   
 

                                                                                 
    


















                                                                                                         
                                                 





                                                                                                                    
    


                                                                                                                           
 


                                             
    
                                           
                                                                                                                                   


                                                                                




                                                                                                       
     
 




                                                                                                           
 
                                                                                                              
 
                                                  
 



                                                                                     
 








                                                                                                                               
 





                                                                      
 








































                                                                                                                                                                              
 
use std::{io::{Error, ErrorKind, Write}, fs::File};

use reqwest::Client;

use futures_util::StreamExt;

use crate::{config::Cfg, modrinth::{projects, Project, versions, extract_current_version, Version}, get_current_list, db::{userlist_get_all_ids, mods_get_versions, userlist_get_applicable_versions, userlist_change_versions}, List};

pub async fn update(config: Cfg) -> Result<(), Box<dyn std::error::Error>> {

    let current_list = get_current_list(config.clone())?;

    let mods = userlist_get_all_ids(config.clone(), current_list.clone().id)?;

    let mut versions = mods_get_versions(config.clone(), mods.clone())?;
    versions.sort_by_key(|ver| ver.mod_id.clone());

    let mut projects = projects(String::from(&config.apis.modrinth), mods).await;
    projects.sort_by_key(|pro| pro.id.clone());
    
    let mut updatestack: Vec<Version> = vec![];
    for (index, project) in projects.into_iter().enumerate() {
        let current_version = &versions[index];

        let p_id = String::from(&project.id);
        let v_id = &current_version.mod_id;

        if &p_id != v_id { return Err(Box::new(Error::new(ErrorKind::Other, "SORTING_ERROR"))) };
        
        if project.versions.join("|") != current_version.versions {
            updatestack.push(match specific_update(config.clone(), current_list.clone(), project).await {
                Ok(ver) => ver,
                //TODO handle errors (only continue on "NO_UPDATE_AVAILABLE")
                Err(_) => { continue; },
            });
        };
    };
    //println!("{:?}", updatestack);

    download_updates(config, updatestack).await?;

    Ok(())
}

async fn specific_update(config: Cfg, list: List, project: Project) -> Result<Version, Box<dyn std::error::Error>> {
    print!("Checking update for '{}' in {}", project.title, list.id);
    
    let applicable_versions = versions(String::from(&config.apis.modrinth), String::from(&project.id), list.clone()).await;
    
    let mut versions: Vec<String> = vec![];

    for ver in &applicable_versions {
        versions.push(String::from(&ver.id));
    }
    
    let mut current: Vec<Version> = vec![];
    if versions.join("|") != userlist_get_applicable_versions(config.clone(), String::from(&list.id), String::from(&project.id))? {
        //get new versions
        print!(" | getting new version");
        let current_str = extract_current_version(applicable_versions.clone())?;
        let current_ver = applicable_versions.into_iter().find(|ver| ver.id == current_str).ok_or("")?;
        current.push(current_ver.clone());

        let link = current_ver.files.into_iter().find(|f| f.primary).ok_or("")?.url;
        userlist_change_versions(config, list.id, current_str, versions.join("|"), link, project.id)?;
    }

    if current.is_empty() { return Err(Box::new(Error::new(ErrorKind::NotFound, "NO_UPDATE_AVAILABLE"))) };
    
    println!(" | ✔️");
    Ok(current[0].clone())
}

async fn download_updates(config: Cfg, versions: Vec<Version>) -> Result<String, Box<dyn std::error::Error>> {

    let dl_path = String::from(&config.downloads);

    for ver in versions {
        let primary_file = ver.files.into_iter().find(|file| file.primary).unwrap();
        let dl_path_file = format!("{}/{}", config.downloads, primary_file.filename);
        println!("Downloading {}", primary_file.url);

        let res = Client::new()
            .get(String::from(&primary_file.url))
            .send()
            .await
            .or(Err(format!("Failed to GET from '{}'", &primary_file.url)))?;
        
        // download chunks
        let mut file = File::create(String::from(&dl_path_file)).or(Err(format!("Failed to create file '{}'", dl_path_file)))?;
        let mut stream = res.bytes_stream();

        while let Some(item) = stream.next().await {
            let chunk = item.or(Err("Error while downloading file"))?;
            file.write_all(&chunk)
                .or(Err("Error while writing to file"))?;
        }
    }

    Ok(dl_path)
}

#[tokio::test]
async fn download_updates_test() {

    use crate::{modrinth::{Version, VersionFile, Hash, VersionType}, config::{Cfg, Apis}};
    
    let config = Cfg { data: "...".to_string(), clean_remove: false, downloads: "./dl".to_string(), apis: Apis { modrinth: "...".to_string() } };

    let versions = vec![Version { 
            id: "dEqtGnT9".to_string(),
            project_id: "kYuIpRLv".to_string(),
            author_id: "Qnt13hO8".to_string(),
            featured: true,
            name: "1.2.2-1.19 - Fabric".to_string(),
            version_number: "1.2.2-1.19".to_string(),
            changelog: None,
            date_published: "2022-11-02T17:41:43.072267Z".to_string(),
            downloads: 58,
            version_type: VersionType::release,
            files: vec![VersionFile {
                    hashes: Hash {
                        sha1: "fdc6dc39427fc92cc1d7ad8b275b5b83325e712b".to_string(),
                        sha512: "5b372f00d6e5d6a5ef225c3897826b9f6a2be5506905f7f71b9e939779765b41be6f2a9b029cfc752ad0751d0d2d5f8bb4544408df1363eebdde15641e99a849".to_string()
                    },
                    url: "https://cdn.modrinth.com/data/kYuIpRLv/versions/dEqtGnT9/waveycapes-fabric-1.2.2-mc1.19.2.jar".to_string(),
                    filename: "waveycapes-fabric-1.2.2-mc1.19.2.jar".to_string(),
                    primary: true,
                    size: 323176
            }],
            game_versions: vec![
                "1.19".to_string(),
                "1.19.1".to_string(),
                "1.19.2".to_string()
            ],
            loaders: vec![
                "fabric".to_string()
            ]
        }];
    assert_eq!(download_updates(config, versions).await.unwrap(), "./dl")
}