Retry connect failures, write plugins to disk instead of zip archive
Writing to the zip was starting to take forever. It makes more sense to just use my big HD and then zip after I'm done downloading every file.
This commit is contained in:
parent
f62324d36c
commit
8a356ac7f5
3
.gitignore
vendored
3
.gitignore
vendored
@ -1,3 +1,4 @@
|
|||||||
/target
|
/target
|
||||||
.env
|
.env
|
||||||
plugins.zip
|
plugins.zip
|
||||||
|
plugins
|
73
src/main.rs
73
src/main.rs
@ -8,18 +8,18 @@ use sqlx::postgres::PgPoolOptions;
|
|||||||
use std::borrow::Borrow;
|
use std::borrow::Borrow;
|
||||||
use std::convert::TryInto;
|
use std::convert::TryInto;
|
||||||
use std::env;
|
use std::env;
|
||||||
use std::fs::OpenOptions;
|
|
||||||
use std::io::Seek;
|
use std::io::Seek;
|
||||||
use std::io::SeekFrom;
|
use std::io::SeekFrom;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::process::Command;
|
use std::process::Command;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use tempfile::tempdir;
|
use tempfile::tempdir;
|
||||||
|
use tokio::fs::create_dir_all;
|
||||||
|
use tokio::io::AsyncWriteExt;
|
||||||
use tokio::io::{AsyncReadExt, AsyncSeekExt};
|
use tokio::io::{AsyncReadExt, AsyncSeekExt};
|
||||||
use tokio::time::sleep;
|
use tokio::time::sleep;
|
||||||
use tracing::{debug, error, info, info_span, warn};
|
use tracing::{debug, error, info, info_span, warn};
|
||||||
use unrar::Archive;
|
use unrar::Archive;
|
||||||
use zip::write::{FileOptions, ZipWriter};
|
|
||||||
|
|
||||||
mod models;
|
mod models;
|
||||||
mod nexus_api;
|
mod nexus_api;
|
||||||
@ -59,17 +59,13 @@ fn get_local_form_id_and_master<'a>(
|
|||||||
Ok((local_form_id, masters[master_index]))
|
Ok((local_form_id, masters[master_index]))
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn process_plugin<W>(
|
async fn process_plugin(
|
||||||
plugin_buf: &mut [u8],
|
plugin_buf: &mut [u8],
|
||||||
pool: &sqlx::Pool<sqlx::Postgres>,
|
pool: &sqlx::Pool<sqlx::Postgres>,
|
||||||
plugin_archive: &mut ZipWriter<W>,
|
|
||||||
db_file: &File,
|
db_file: &File,
|
||||||
mod_obj: &Mod,
|
db_mod: &Mod,
|
||||||
file_path: &str,
|
file_path: &str,
|
||||||
) -> Result<()>
|
) -> Result<()> {
|
||||||
where
|
|
||||||
W: std::io::Write + std::io::Seek,
|
|
||||||
{
|
|
||||||
if plugin_buf.len() == 0 {
|
if plugin_buf.len() == 0 {
|
||||||
warn!("skipping processing of invalid empty plugin");
|
warn!("skipping processing of invalid empty plugin");
|
||||||
return Ok(());
|
return Ok(());
|
||||||
@ -174,31 +170,19 @@ where
|
|||||||
warn!(error = %err, "Failed to parse plugin, skipping plugin");
|
warn!(error = %err, "Failed to parse plugin, skipping plugin");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
plugin_archive.start_file(
|
|
||||||
format!(
|
|
||||||
"{}/{}/{}/{}",
|
|
||||||
GAME_NAME, mod_obj.nexus_mod_id, db_file.nexus_file_id, file_path
|
|
||||||
),
|
|
||||||
FileOptions::default(),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let mut reader = std::io::Cursor::new(&plugin_buf);
|
let plugin_path = format!(
|
||||||
std::io::copy(&mut reader, plugin_archive)?;
|
"plugins/{}/{}/{}/{}",
|
||||||
Ok(())
|
GAME_NAME, db_mod.nexus_mod_id, db_file.nexus_file_id, file_path
|
||||||
}
|
|
||||||
|
|
||||||
fn initialize_plugins_archive(mod_id: i32, file_id: i32) -> Result<()> {
|
|
||||||
let mut plugins_archive = ZipWriter::new(
|
|
||||||
OpenOptions::new()
|
|
||||||
.write(true)
|
|
||||||
.create(true)
|
|
||||||
.open("plugins.zip")?,
|
|
||||||
);
|
);
|
||||||
plugins_archive.add_directory(
|
let plugin_path = Path::new(&plugin_path);
|
||||||
format!("{}/{}/{}", GAME_NAME, mod_id, file_id),
|
if let Some(dir) = plugin_path.parent() {
|
||||||
FileOptions::default(),
|
create_dir_all(dir).await?;
|
||||||
)?;
|
}
|
||||||
plugins_archive.finish()?;
|
let mut file = tokio::fs::File::create(plugin_path).await?;
|
||||||
|
|
||||||
|
info!(path = %plugin_path.display(), "saving plugin to disk");
|
||||||
|
file.write_all(&plugin_buf).await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -331,13 +315,11 @@ pub async fn main() -> Result<()> {
|
|||||||
let mut tokio_file = download_link_resp.download_file(&client).await?;
|
let mut tokio_file = download_link_resp.download_file(&client).await?;
|
||||||
info!(bytes = api_file.size, "download finished");
|
info!(bytes = api_file.size, "download finished");
|
||||||
|
|
||||||
initialize_plugins_archive(db_mod.nexus_mod_id, db_file.nexus_file_id)?;
|
create_dir_all(format!(
|
||||||
let mut plugins_archive = ZipWriter::new_append(
|
"plugins/{}/{}/{}",
|
||||||
OpenOptions::new()
|
GAME_NAME, db_mod.nexus_mod_id, db_file.nexus_file_id
|
||||||
.read(true)
|
))
|
||||||
.write(true)
|
.await?;
|
||||||
.open("plugins.zip")?,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let mut initial_bytes = [0; 8];
|
let mut initial_bytes = [0; 8];
|
||||||
tokio_file.seek(SeekFrom::Start(0)).await?;
|
tokio_file.seek(SeekFrom::Start(0)).await?;
|
||||||
@ -417,7 +399,6 @@ pub async fn main() -> Result<()> {
|
|||||||
process_plugin(
|
process_plugin(
|
||||||
&mut plugin_buf,
|
&mut plugin_buf,
|
||||||
&pool,
|
&pool,
|
||||||
&mut plugins_archive,
|
|
||||||
&db_file,
|
&db_file,
|
||||||
&db_mod,
|
&db_mod,
|
||||||
&file_path.to_string_lossy(),
|
&file_path.to_string_lossy(),
|
||||||
@ -498,7 +479,6 @@ pub async fn main() -> Result<()> {
|
|||||||
process_plugin(
|
process_plugin(
|
||||||
&mut plugin_buf,
|
&mut plugin_buf,
|
||||||
&pool,
|
&pool,
|
||||||
&mut plugins_archive,
|
|
||||||
&db_file,
|
&db_file,
|
||||||
&db_mod,
|
&db_mod,
|
||||||
file_path,
|
file_path,
|
||||||
@ -511,20 +491,11 @@ pub async fn main() -> Result<()> {
|
|||||||
Err(err)
|
Err(err)
|
||||||
}
|
}
|
||||||
}?;
|
}?;
|
||||||
process_plugin(
|
process_plugin(&mut buf, &pool, &db_file, &db_mod, file_path).await?;
|
||||||
&mut buf,
|
|
||||||
&pool,
|
|
||||||
&mut plugins_archive,
|
|
||||||
&db_file,
|
|
||||||
&db_mod,
|
|
||||||
file_path,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
plugins_archive.finish()?;
|
|
||||||
debug!(duration = ?download_link_resp.wait, "sleeping");
|
debug!(duration = ?download_link_resp.wait, "sleeping");
|
||||||
sleep(download_link_resp.wait).await;
|
sleep(download_link_resp.wait).await;
|
||||||
}
|
}
|
||||||
|
@ -5,11 +5,10 @@ use serde_json::Value;
|
|||||||
use std::{env, time::Duration};
|
use std::{env, time::Duration};
|
||||||
use tempfile::tempfile;
|
use tempfile::tempfile;
|
||||||
use tokio::fs::File;
|
use tokio::fs::File;
|
||||||
use tokio::time::sleep;
|
|
||||||
use tokio_util::compat::FuturesAsyncReadCompatExt;
|
use tokio_util::compat::FuturesAsyncReadCompatExt;
|
||||||
use tracing::{info, instrument, warn};
|
use tracing::{info, instrument};
|
||||||
|
|
||||||
use super::{rate_limit_wait_duration, GAME_NAME, USER_AGENT};
|
use super::{rate_limit_wait_duration, warn_and_sleep, GAME_NAME, USER_AGENT};
|
||||||
|
|
||||||
pub struct DownloadLinkResponse {
|
pub struct DownloadLinkResponse {
|
||||||
pub wait: Duration,
|
pub wait: Duration,
|
||||||
@ -28,13 +27,17 @@ pub async fn get(client: &Client, mod_id: i32, file_id: i64) -> Result<DownloadL
|
|||||||
.header("apikey", env::var("NEXUS_API_KEY")?)
|
.header("apikey", env::var("NEXUS_API_KEY")?)
|
||||||
.header("user-agent", USER_AGENT)
|
.header("user-agent", USER_AGENT)
|
||||||
.send()
|
.send()
|
||||||
.await?
|
.await
|
||||||
.error_for_status()
|
|
||||||
{
|
{
|
||||||
Ok(res) => res,
|
Ok(res) => match res.error_for_status() {
|
||||||
|
Ok(res) => res,
|
||||||
|
Err(err) => {
|
||||||
|
warn_and_sleep("download_link::get", anyhow!(err), attempt).await;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
},
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
warn!(error = %err, attempt, "Failed to get download link for file, trying again after 1 second");
|
warn_and_sleep("download_link::get", anyhow!(err), attempt).await;
|
||||||
sleep(std::time::Duration::from_secs(1)).await;
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -69,13 +72,25 @@ impl DownloadLinkResponse {
|
|||||||
pub async fn download_file(&self, client: &Client) -> Result<File> {
|
pub async fn download_file(&self, client: &Client) -> Result<File> {
|
||||||
for attempt in 1..=3 {
|
for attempt in 1..=3 {
|
||||||
let mut tokio_file = File::from_std(tempfile()?);
|
let mut tokio_file = File::from_std(tempfile()?);
|
||||||
let res = client
|
let res = match client
|
||||||
.get(self.link()?)
|
.get(self.link()?)
|
||||||
.header("apikey", env::var("NEXUS_API_KEY")?)
|
.header("apikey", env::var("NEXUS_API_KEY")?)
|
||||||
.header("user-agent", USER_AGENT)
|
.header("user-agent", USER_AGENT)
|
||||||
.send()
|
.send()
|
||||||
.await?
|
.await
|
||||||
.error_for_status()?;
|
{
|
||||||
|
Ok(res) => match res.error_for_status() {
|
||||||
|
Ok(res) => res,
|
||||||
|
Err(err) => {
|
||||||
|
warn_and_sleep("download_link::download_file", anyhow!(err), attempt).await;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(err) => {
|
||||||
|
warn_and_sleep("download_link::download_file", anyhow!(err), attempt).await;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
};
|
||||||
info!(status = %res.status(), "downloading file from nexus");
|
info!(status = %res.status(), "downloading file from nexus");
|
||||||
|
|
||||||
// See: https://github.com/benkay86/async-applied/blob/master/reqwest-tokio-compat/src/main.rs
|
// See: https://github.com/benkay86/async-applied/blob/master/reqwest-tokio-compat/src/main.rs
|
||||||
@ -90,8 +105,7 @@ impl DownloadLinkResponse {
|
|||||||
return Ok(tokio_file);
|
return Ok(tokio_file);
|
||||||
}
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
warn!(error = %err, attempt, "Failed to download file, trying again after 1 second");
|
warn_and_sleep("download_link::download_file", anyhow!(err), attempt).await
|
||||||
sleep(std::time::Duration::from_secs(1)).await;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,10 +3,9 @@ use chrono::NaiveDateTime;
|
|||||||
use reqwest::Client;
|
use reqwest::Client;
|
||||||
use serde_json::Value;
|
use serde_json::Value;
|
||||||
use std::{env, time::Duration};
|
use std::{env, time::Duration};
|
||||||
use tokio::time::sleep;
|
use tracing::{info, instrument};
|
||||||
use tracing::{info, instrument, warn};
|
|
||||||
|
|
||||||
use super::{rate_limit_wait_duration, GAME_NAME, USER_AGENT};
|
use super::{rate_limit_wait_duration, warn_and_sleep, GAME_NAME, USER_AGENT};
|
||||||
|
|
||||||
pub struct FilesResponse {
|
pub struct FilesResponse {
|
||||||
pub wait: Duration,
|
pub wait: Duration,
|
||||||
@ -37,13 +36,17 @@ pub async fn get(client: &Client, nexus_mod_id: i32) -> Result<FilesResponse> {
|
|||||||
.header("apikey", env::var("NEXUS_API_KEY")?)
|
.header("apikey", env::var("NEXUS_API_KEY")?)
|
||||||
.header("user-agent", USER_AGENT)
|
.header("user-agent", USER_AGENT)
|
||||||
.send()
|
.send()
|
||||||
.await?
|
.await
|
||||||
.error_for_status()
|
|
||||||
{
|
{
|
||||||
Ok(res) => res,
|
Ok(res) => match res.error_for_status() {
|
||||||
|
Ok(res) => res,
|
||||||
|
Err(err) => {
|
||||||
|
warn_and_sleep("files::get", anyhow!(err), attempt).await;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
},
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
warn!(error = %err, attempt, "Failed to get files for mod, trying again after 1 second");
|
warn_and_sleep("files::get", anyhow!(err), attempt).await;
|
||||||
sleep(std::time::Duration::from_secs(1)).await;
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -2,11 +2,10 @@ use anyhow::{anyhow, Result};
|
|||||||
use reqwest::Client;
|
use reqwest::Client;
|
||||||
use serde_json::Value;
|
use serde_json::Value;
|
||||||
use std::env;
|
use std::env;
|
||||||
use tokio::time::sleep;
|
use tracing::{info, instrument};
|
||||||
use tracing::{info, instrument, warn};
|
|
||||||
|
|
||||||
use super::files::ApiFile;
|
use super::files::ApiFile;
|
||||||
use super::USER_AGENT;
|
use super::{warn_and_sleep, USER_AGENT};
|
||||||
|
|
||||||
fn has_plugin(json: &Value) -> Result<bool> {
|
fn has_plugin(json: &Value) -> Result<bool> {
|
||||||
let node_type = json
|
let node_type = json
|
||||||
@ -53,13 +52,17 @@ pub async fn contains_plugin(client: &Client, api_file: &ApiFile<'_>) -> Result<
|
|||||||
.header("apikey", env::var("NEXUS_API_KEY")?)
|
.header("apikey", env::var("NEXUS_API_KEY")?)
|
||||||
.header("user-agent", USER_AGENT)
|
.header("user-agent", USER_AGENT)
|
||||||
.send()
|
.send()
|
||||||
.await?
|
.await
|
||||||
.error_for_status()
|
|
||||||
{
|
{
|
||||||
Ok(res) => res,
|
Ok(res) => match res.error_for_status() {
|
||||||
|
Ok(res) => res,
|
||||||
|
Err(err) => {
|
||||||
|
warn_and_sleep("metadata::contains_plugin", anyhow!(err), attempt).await;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
},
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
warn!(error = %err, attempt, "Failed to get metadata for file, trying again after 1 second");
|
warn_and_sleep("metadata::contains_plugin", anyhow!(err), attempt).await;
|
||||||
sleep(std::time::Duration::from_secs(1)).await;
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -3,7 +3,8 @@ use chrono::DateTime;
|
|||||||
use chrono::Duration;
|
use chrono::Duration;
|
||||||
use chrono::Utc;
|
use chrono::Utc;
|
||||||
use reqwest::Response;
|
use reqwest::Response;
|
||||||
use tracing::info;
|
use tokio::time::sleep;
|
||||||
|
use tracing::{info, warn};
|
||||||
|
|
||||||
pub mod download_link;
|
pub mod download_link;
|
||||||
pub mod files;
|
pub mod files;
|
||||||
@ -51,3 +52,8 @@ pub fn rate_limit_wait_duration(res: &Response) -> Result<std::time::Duration> {
|
|||||||
Ok(std::time::Duration::from_secs(1))
|
Ok(std::time::Duration::from_secs(1))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn warn_and_sleep(request_name: &str, err: anyhow::Error, attempt: i32) {
|
||||||
|
warn!(error = %err, attempt, "{} request failed, trying again after 1 second", request_name);
|
||||||
|
sleep(std::time::Duration::from_secs(1)).await;
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user