forked from Hirevo/alexandrie
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Issue Hirevo#183 : add categorie pages
Add new pages to list all categores. It is possible to go here using the "Categorie" button or using links in each crate's view
- Loading branch information
1 parent
46c26af
commit 96fb7b9
Showing
8 changed files
with
819 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,156 @@ | ||
use std::num::NonZeroU32; | ||
use std::sync::Arc; | ||
|
||
use axum::extract::{Path, State, Query}; | ||
use axum::response::Redirect; | ||
use axum_extra::either::Either; | ||
use axum_extra::response::Html; | ||
use diesel::prelude::*; | ||
use diesel::dsl as sql; | ||
|
||
use json::json; | ||
use serde::{Deserialize, Serialize}; | ||
|
||
use alexandrie_index::Indexer; | ||
|
||
use crate::config::AppState; | ||
use crate::db::models::Crate; | ||
use crate::db::schema::*; | ||
use crate::db::DATETIME_FORMAT; | ||
use crate::error::{Error, FrontendError}; | ||
use crate::frontend::helpers; | ||
use crate::utils::auth::frontend::Auth; | ||
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] | ||
pub(crate) struct QueryParams { | ||
pub page: Option<NonZeroU32>, | ||
} | ||
|
||
fn paginated_url(categorie: &str, page_number: u32, page_count: u32) -> Option<String> { | ||
if page_number >= 1 && page_number <= page_count { | ||
Some(format!("/categories/{0}?page={1}", categorie, page_number)) | ||
} else { | ||
None | ||
} | ||
} | ||
|
||
pub(crate) async fn get( | ||
State(state): State<Arc<AppState>>, | ||
Query(params): Query<QueryParams>, | ||
Path(categorie): Path<String>, | ||
user: Option<Auth>, | ||
) -> Result<Either<Html<String>, Redirect>, FrontendError> { | ||
let page_number = params.page.map_or_else(|| 1, |page| page.get()); | ||
|
||
if state.is_login_required() && user.is_none() { | ||
return Ok(Either::E2(Redirect::to("/account/login"))); | ||
} | ||
|
||
let db: &crate::db::database::Database<SqliteConnection> = &state.db; | ||
let state = Arc::clone(&state); | ||
|
||
let transaction = db.transaction(move |conn| { | ||
|
||
//? Get the total count of search results. | ||
let total_results = crate_categories::table | ||
.inner_join(categories::table) | ||
.inner_join(crates::table) | ||
.filter(categories::name.eq(&categorie)) | ||
.select(sql::count(crates::id)) | ||
.first::<i64>(conn)?; | ||
|
||
//? Get the search results for the given page number. | ||
//? First get all ids of crates with given categories | ||
let results = crate_categories::table | ||
.inner_join(categories::table) | ||
.inner_join(crates::table) | ||
.filter(categories::name.eq(&categorie)) | ||
.select(crates::id) | ||
.order_by(crates::downloads.desc()) | ||
.limit(15) | ||
.offset(15 * i64::from(page_number - 1)) | ||
.load::<i64>(conn)?; | ||
|
||
let results = results | ||
.into_iter() | ||
.map(|crate_ids| { | ||
let categories = crate_categories::table | ||
.inner_join(categories::table) | ||
.select(categories::name) | ||
.filter(crate_categories::crate_id.eq(crate_ids)) | ||
.load::<String>(conn)?; | ||
|
||
let categorie_crate: Crate = crates::table | ||
.filter(crates::id.eq(crate_ids)) | ||
.first(conn)?; | ||
|
||
Ok((categorie_crate, categories)) | ||
}) | ||
.collect::<Result<Vec<(Crate, Vec<String>)>, Error>>()?; | ||
|
||
let description = categories::table | ||
.filter(categories::name.eq(&categorie)) | ||
.select(categories::description) | ||
.first(conn); | ||
|
||
let description : String = match description { | ||
Err(_e) => "".to_string(), | ||
Ok(s) => s, | ||
}; | ||
|
||
//? Make page number starts counting from 1 (instead of 0). | ||
let page_count = (total_results / 15 | ||
+ if total_results > 0 && total_results % 15 == 0 { | ||
0 | ||
} else { | ||
1 | ||
}) as u32; | ||
|
||
let next_page = paginated_url(&categorie, page_number + 1, page_count); | ||
let prev_page = paginated_url(&categorie, page_number - 1, page_count); | ||
|
||
let auth = &state.frontend.config.auth; | ||
let engine = &state.frontend.handlebars; | ||
let context = json!({ | ||
"auth_disabled": !auth.enabled(), | ||
"registration_disabled": !auth.allow_registration(), | ||
"name": &categorie, | ||
"description": description, | ||
"user": user.map(|it| it.into_inner()), | ||
"instance": &state.frontend.config, | ||
"total_results": total_results, | ||
"pagination": { | ||
"current": page_number, | ||
"total_count": page_count, | ||
"next": next_page, | ||
"prev": prev_page, | ||
}, | ||
"results": results.into_iter().map(|(krate, categories)| { | ||
let record = state.index.latest_record(&krate.name)?; | ||
let created_at = | ||
chrono::NaiveDateTime::parse_from_str(krate.created_at.as_str(), DATETIME_FORMAT) | ||
.unwrap(); | ||
let updated_at = | ||
chrono::NaiveDateTime::parse_from_str(krate.updated_at.as_str(), DATETIME_FORMAT) | ||
.unwrap(); | ||
Ok(json!({ | ||
"id": krate.id, | ||
"name": krate.name, | ||
"version": record.vers, | ||
"description": krate.description, | ||
"created_at": helpers::humanize_datetime(created_at), | ||
"updated_at": helpers::humanize_datetime(updated_at), | ||
"downloads": helpers::humanize_number(krate.downloads), | ||
"documentation": krate.documentation, | ||
"repository": krate.repository, | ||
"categories": categories, | ||
"yanked": record.yanked, | ||
})) | ||
}).collect::<Result<Vec<_>, Error>>()?, | ||
}); | ||
Ok(Either::E1(Html( | ||
engine.render("categories", &context).unwrap(), | ||
))) | ||
}); | ||
|
||
transaction.await | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,83 @@ | ||
use std::sync::Arc; | ||
|
||
use axum::extract::State; | ||
use axum::response::Redirect; | ||
use axum_extra::either::Either; | ||
use axum_extra::response::Html; | ||
use diesel::dsl as sql; | ||
use diesel::prelude::*; | ||
|
||
use json::json; | ||
|
||
use crate::config::AppState; | ||
use crate::db::models::Crate; | ||
use crate::db::schema::*; | ||
use crate::error::{Error, FrontendError}; | ||
use crate::utils::auth::frontend::Auth; | ||
|
||
pub(crate) async fn get( | ||
State(state): State<Arc<AppState>>, | ||
user: Option<Auth>, | ||
) -> Result<Either<Html<String>, Redirect>, FrontendError> { | ||
if state.is_login_required() && user.is_none() { | ||
return Ok(Either::E2(Redirect::to("/account/login"))); | ||
} | ||
|
||
let db: &crate::db::database::Database<SqliteConnection> = &state.db; | ||
let state = Arc::clone(&state); | ||
|
||
let transaction = db.transaction(move |conn| { | ||
//? Get the total count of search results. | ||
let total_results = categories::table | ||
.select(sql::count(categories::id)) | ||
.first::<i64>(conn)?; | ||
|
||
let categories = categories::table | ||
.select((categories::name, categories::description)) | ||
.load::<(String, String)>(conn)?; | ||
|
||
let auth = &state.frontend.config.auth; | ||
let engine = &state.frontend.handlebars; | ||
let context = json!({ | ||
"auth_disabled": !auth.enabled(), | ||
"registration_disabled": !auth.allow_registration(), | ||
"user": user.map(|it| it.into_inner()), | ||
"instance": &state.frontend.config, | ||
"total_results": total_results, | ||
"categories": categories.into_iter().map(|categorie| { | ||
let crates = crate_categories::table | ||
.inner_join(categories::table) | ||
.inner_join(crates::table) | ||
.filter(categories::name.eq(&categorie.0)) | ||
.order_by(crates::downloads.desc()) | ||
.select(crates::id) | ||
.limit(10) | ||
.load::<i64>(conn)? | ||
.into_iter() | ||
.map(|crate_ids| { | ||
let categorie_crate: Crate = crates::table | ||
.filter(crates::id.eq(crate_ids)) | ||
.first(conn)?; | ||
Ok(categorie_crate) | ||
}) | ||
.collect::<Result<Vec<Crate>, Error>>()?; | ||
let count = crate_categories::table | ||
.inner_join(categories::table) | ||
.inner_join(crates::table) | ||
.filter(categories::name.eq(&categorie.0)) | ||
.select(sql::count(crates::id)) | ||
.first::<i64>(conn)?; | ||
Ok(json!({ | ||
"name":&categorie.0, | ||
"description":&categorie.1, | ||
"crates": crates, | ||
"count": count | ||
}))}).collect::<Result<Vec<_>, Error>>()?, | ||
}); | ||
Ok(Either::E1(Html( | ||
engine.render("categories_index", &context).unwrap(), | ||
))) | ||
}); | ||
|
||
transaction.await | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.