diff --git a/plume-models/src/admin.rs b/plume-models/src/admin.rs index a4fa04551..37a345842 100644 --- a/plume-models/src/admin.rs +++ b/plume-models/src/admin.rs @@ -5,7 +5,7 @@ use rocket::{ Outcome, }; -/// Wrapper around User to use as a request guard on pages reserved to admins. +/// Wrapper around User to use as a request guard on pages exclusively reserved to admins. pub struct Admin(pub User); impl<'a, 'r> FromRequest<'a, 'r> for Admin { @@ -21,6 +21,23 @@ impl<'a, 'r> FromRequest<'a, 'r> for Admin { } } +/// Same as `Admin` but it forwards to next guard if the user is not an admin. +/// It's useful when there are multiple implementations of routes for admin and moderator. +pub struct InclusiveAdmin(pub User); + +impl<'a, 'r> FromRequest<'a, 'r> for InclusiveAdmin { + type Error = (); + + fn from_request(request: &'a Request<'r>) -> request::Outcome { + let user = request.guard::()?; + if user.is_admin() { + Outcome::Success(InclusiveAdmin(user)) + } else { + Outcome::Forward(()) + } + } +} + /// Same as `Admin` but for moderators. pub struct Moderator(pub User); diff --git a/src/routes/instance.rs b/src/routes/instance.rs index 9de9fee02..e05e069ca 100644 --- a/src/routes/instance.rs +++ b/src/routes/instance.rs @@ -51,7 +51,7 @@ pub fn index(conn: DbConn, rockets: PlumeRocket) -> Result { } #[get("/admin")] -pub fn admin(_admin: Admin, conn: DbConn, rockets: PlumeRocket) -> Result { +pub fn admin(_admin: InclusiveAdmin, conn: DbConn, rockets: PlumeRocket) -> Result { let local_inst = Instance::get_local()?; Ok(render!(instance::admin( &(&conn, &rockets).to_context(), diff --git a/templates/base.rs.html b/templates/base.rs.html index 564c1f84f..711ed2387 100644 --- a/templates/base.rs.html +++ b/templates/base.rs.html @@ -87,6 +87,8 @@

@Instance::get_local().map(|i| i.name).unwrap_or_default()

@i18n!(ctx.1, "Privacy policy") @if ctx.2.clone().map(|u| u.is_admin()).unwrap_or(false) { @i18n!(ctx.1, "Administration") + } else if ctx.2.clone().map(|u| u.is_moderator()).unwrap_or(false) { + @i18n!(ctx.1, "Moderation") }
diff --git a/templates/instance/admin.rs.html b/templates/instance/admin.rs.html index 7ff3750d2..8261bb1a4 100644 --- a/templates/instance/admin.rs.html +++ b/templates/instance/admin.rs.html @@ -1,6 +1,6 @@ @use plume_models::instance::Instance; @use validator::ValidationErrors; -@use crate::templates::base; +@use crate::templates::{base, instance::admin_header}; @use crate::template_utils::*; @use crate::routes::instance::InstanceSettingsForm; @use crate::routes::*; @@ -8,14 +8,7 @@ @(ctx: BaseContext, instance: Instance, form: InstanceSettingsForm, errors: ValidationErrors) @:base(ctx, i18n!(ctx.1, "Administration of {0}"; instance.name.clone()), {}, {}, { -

@i18n!(ctx.1, "Administration")

- - @tabs(&[ - (&uri!(instance::admin).to_string(), i18n!(ctx.1, "Configuration"), true), - (&uri!(instance::admin_instances: page = _).to_string(), i18n!(ctx.1, "Instances"), false), - (&uri!(instance::admin_users: page = _).to_string(), i18n!(ctx.1, "Users"), false), - (&uri!(instance::admin_email_blocklist: page=_).to_string(), i18n!(ctx.1, "Email blocklist"), false) - ]) + @:admin_header(ctx, "Administration", 1)
@(Input::new("name", i18n!(ctx.1, "Name")) diff --git a/templates/instance/admin_header.rs.html b/templates/instance/admin_header.rs.html new file mode 100644 index 000000000..7794f39f4 --- /dev/null +++ b/templates/instance/admin_header.rs.html @@ -0,0 +1,21 @@ +@use crate::template_utils::*; +@use crate::routes::*; + +@(ctx: BaseContext, title: &str, selected_tab: u8) + +

@i18n!(ctx.1, title)

+ +@if ctx.2.clone().map(|u| u.is_admin()).unwrap_or(false) { + @tabs(&[ + (&uri!(instance::admin).to_string(), i18n!(ctx.1, "Configuration"), selected_tab == 1), + (&uri!(instance::admin_instances: page = _).to_string(), i18n!(ctx.1, "Instances"), selected_tab == 2), + (&uri!(instance::admin_users: page = _).to_string(), i18n!(ctx.1, "Users"), selected_tab == 3), + (&uri!(instance::admin_email_blocklist: page=_).to_string(), i18n!(ctx.1, "Email blocklist"), selected_tab == 4) + ]) +} else { + @tabs(&[ + (&uri!(instance::admin_instances: page = _).to_string(), i18n!(ctx.1, "Instances"), selected_tab == 2), + (&uri!(instance::admin_users: page = _).to_string(), i18n!(ctx.1, "Users"), selected_tab == 3), + (&uri!(instance::admin_email_blocklist: page=_).to_string(), i18n!(ctx.1, "Email blocklist"), selected_tab == 4) + ]) +} diff --git a/templates/instance/admin_mod.rs.html b/templates/instance/admin_mod.rs.html index e215d7e1e..fc1cb0a91 100644 --- a/templates/instance/admin_mod.rs.html +++ b/templates/instance/admin_mod.rs.html @@ -1,15 +1,8 @@ -@use crate::templates::base; +@use crate::templates::{base, instance::admin_header}; @use crate::template_utils::*; -@use crate::routes::*; @(ctx: BaseContext) @:base(ctx, i18n!(ctx.1, "Moderation"), {}, {}, { -

@i18n!(ctx.1, "Moderation")

- - @tabs(&[ - (&uri!(instance::admin).to_string(), i18n!(ctx.1, "Home"), true), - (&uri!(instance::admin_instances: page = _).to_string(), i18n!(ctx.1, "Instances"), false), - (&uri!(instance::admin_users: page = _).to_string(), i18n!(ctx.1, "Users"), false), - ]) + @:admin_header(ctx, "Moderation", 0) }) diff --git a/templates/instance/emailblocklist.rs.html b/templates/instance/emailblocklist.rs.html index fe9729fcc..7e82cea4e 100644 --- a/templates/instance/emailblocklist.rs.html +++ b/templates/instance/emailblocklist.rs.html @@ -1,17 +1,11 @@ @use plume_models::blocklisted_emails::BlocklistedEmail; -@use crate::templates::base; +@use crate::templates::{base, instance::admin_header}; @use crate::template_utils::*; @use crate::routes::*; @(ctx:BaseContext, emails: Vec, page:i32, n_pages:i32) - @:base(ctx, i18n!(ctx.1, "Blocklisted Emails"), {}, {}, { -

@i18n!(ctx.1,"Blocklisted Emails")

- @tabs(&[ - (&uri!(instance::admin).to_string(), i18n!(ctx.1, "Configuration"), false), - (&uri!(instance::admin_instances: page = _).to_string(), i18n!(ctx.1, "Instances"), false), - (&uri!(instance::admin_users: page = _).to_string(), i18n!(ctx.1, "Users"), false), - (&uri!(instance::admin_email_blocklist:page=_).to_string(), i18n!(ctx.1, "Email blocklist"), true), - ]) +@:base(ctx, i18n!(ctx.1, "Blocklisted Emails"), {}, {}, { + @:admin_header(ctx, "Blocklisted Emails", 4) @(Input::new("email_address", i18n!(ctx.1, "Email address")) .details(i18n!(ctx.1, "The email address you wish to block. In order to block domains, you can use globbing syntax, for example '*@example.com' blocks all addresses from example.com")) diff --git a/templates/instance/list.rs.html b/templates/instance/list.rs.html index 619689f78..baeb11a06 100644 --- a/templates/instance/list.rs.html +++ b/templates/instance/list.rs.html @@ -1,19 +1,12 @@ @use plume_models::instance::Instance; -@use crate::templates::base; +@use crate::templates::{base, instance::admin_header}; @use crate::template_utils::*; @use crate::routes::*; @(ctx: BaseContext, instance: Instance, instances: Vec, page: i32, n_pages: i32) @:base(ctx, i18n!(ctx.1, "Administration of {0}"; instance.name), {}, {}, { -

@i18n!(ctx.1, "Instances")

- - @tabs(&[ - (&uri!(instance::admin).to_string(), i18n!(ctx.1, "Configuration"), false), - (&uri!(instance::admin_instances: page = _).to_string(), i18n!(ctx.1, "Instances"), true), - (&uri!(instance::admin_users: page = _).to_string(), i18n!(ctx.1, "Users"), false), - (&uri!(instance::admin_email_blocklist:page=_).to_string(), i18n!(ctx.1, "Email blocklist"), false), - ]) + @:admin_header(ctx, "Instances", 2))
@for instance in instances { diff --git a/templates/instance/users.rs.html b/templates/instance/users.rs.html index 1499fb527..38299f373 100644 --- a/templates/instance/users.rs.html +++ b/templates/instance/users.rs.html @@ -1,19 +1,12 @@ @use plume_models::users::User; -@use crate::templates::base; +@use crate::templates::{base, instance::admin_header}; @use crate::template_utils::*; @use crate::routes::*; @(ctx: BaseContext, users: Vec, user: Option<&str>, page: i32, n_pages: i32) @:base(ctx, i18n!(ctx.1, "Users"), {}, {}, { -

@i18n!(ctx.1, "Users")

- - @tabs(&[ - (&uri!(instance::admin).to_string(), i18n!(ctx.1, "Configuration"), false), - (&uri!(instance::admin_instances: page = _).to_string(), i18n!(ctx.1, "Instances"), false), - (&uri!(instance::admin_users: page = _).to_string(), i18n!(ctx.1, "Users"), true), - (&uri!(instance::admin_email_blocklist: page=_).to_string(), i18n!(ctx.1, "Email blocklist"), false) - ]) + @:admin_header(ctx, "Users", 3))