Files
AstralRinth/src/routes/updates.rs
2023-01-27 19:24:40 -07:00

105 lines
2.8 KiB
Rust

use std::collections::HashMap;
use actix_web::{get, web, HttpRequest, HttpResponse};
use serde::Serialize;
use sqlx::PgPool;
use crate::database;
use crate::models::projects::{Version, VersionType};
use crate::util::auth::{
get_user_from_headers, is_authorized, is_authorized_version,
};
use futures::StreamExt;
use super::ApiError;
#[get("{id}/forge_updates.json")]
pub async fn forge_updates(
req: HttpRequest,
info: web::Path<(String,)>,
pool: web::Data<PgPool>,
) -> Result<HttpResponse, ApiError> {
const ERROR: &str = "The specified project does not exist!";
let (id,) = info.into_inner();
let project =
database::models::Project::get_from_slug_or_project_id(&id, &**pool)
.await?
.ok_or_else(|| ApiError::InvalidInput(ERROR.to_string()))?;
let user_option = get_user_from_headers(req.headers(), &**pool).await.ok();
if !is_authorized(&project, &user_option, &pool).await? {
return Err(ApiError::InvalidInput(ERROR.to_string()));
}
let version_ids = database::models::Version::get_project_versions(
project.id,
None,
Some(vec!["forge".to_string()]),
None,
None,
None,
&**pool,
)
.await?;
let versions =
database::models::Version::get_many_full(version_ids, &**pool).await?;
let mut versions = futures::stream::iter(versions)
.filter_map(|data| async {
if is_authorized_version(&data.inner, &user_option, &pool)
.await
.ok()?
{
Some(data)
} else {
None
}
})
.collect::<Vec<_>>()
.await;
versions
.sort_by(|a, b| b.inner.date_published.cmp(&a.inner.date_published));
#[derive(Serialize)]
struct ForgeUpdates {
homepage: String,
promos: HashMap<String, String>,
}
let mut response = ForgeUpdates {
homepage: format!(
"{}/mod/{}",
dotenvy::var("SITE_URL").unwrap_or_default(),
id
),
promos: HashMap::new(),
};
for version in versions {
let version = Version::from(version);
if version.version_type == VersionType::Release {
for game_version in &version.game_versions {
response
.promos
.entry(format!("{}-recommended", game_version.0))
.or_insert_with(|| version.version_number.clone());
}
}
for game_version in &version.game_versions {
response
.promos
.entry(format!("{}-latest", game_version.0))
.or_insert_with(|| version.version_number.clone());
}
}
Ok(HttpResponse::Ok().json(response))
}