aboutsummaryrefslogtreecommitdiff
path: root/ui/src/admin/mod.rs
blob: 74a5e1afb1361d1f7eee3c9b0f0d95a7e43b649b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
/*
    This file is part of jellything (https://codeberg.org/metamuffin/jellything)
    which is licensed under the GNU Affero General Public License (version 3); see /COPYING.
    Copyright (C) 2025 metamuffin <metamuffin.org>
*/

pub mod user;
pub mod log;

use crate::{Page, locale::Language, scaffold::FlashDisplay};
use jellycommon::routes::{
    u_admin_import, u_admin_invite_create, u_admin_invite_remove, u_admin_log,
    u_admin_transcode_posters, u_admin_update_search, u_admin_users,
};

impl Page for AdminDashboardPage<'_> {
    fn title(&self) -> String {
        "Admin Dashboard".to_string()
    }
    fn to_render(&self) -> markup::DynRender {
        markup::new!(@self)
    }
}

markup::define!(
    AdminDashboardPage<'a>(lang: &'a Language, busy: Option<&'static str>, last_import_err: &'a [String], flash: Option<Result<String, String>>, invites: &'a [String]) {
        h1 { "Admin Panel" }
        @FlashDisplay { flash: flash.clone() }
        @if !last_import_err.is_empty() {
            section.message.error {
                details {
                    summary { p.error { @format!("The last import resulted in {} errors:", last_import_err.len()) } }
                    ol { @for e in *last_import_err {
                        li.error { pre.error { @e } }
                    }}
                }
            }
        }
        ul {
            li{a[href=u_admin_log(true)] { "Server Log (Warnings only)" }}
            li{a[href=u_admin_log(false)] { "Server Log (Full) " }}
        }
        h2 { "Library" }
        @if let Some(text) = busy {
            section.message { p.warn { @text } }
        }
        form[method="POST", action=u_admin_import(true)] {
            input[type="submit", disabled=busy.is_some(), value="Start incremental import"];
        }
        form[method="POST", action=u_admin_import(false)] {
            input[type="submit", disabled=busy.is_some(), value="Start full import"];
        }
        form[method="POST", action=u_admin_transcode_posters()] {
            input[type="submit", disabled=busy.is_some(), value="Transcode all posters with low resolution"];
        }
        form[method="POST", action=u_admin_update_search()] {
            input[type="submit", value="Regenerate full-text search index"];
        }
        h2 { "Users" }
        p { a[href=u_admin_users()] "Manage Users" }
        h2 { "Invitations" }
        form[method="POST", action=u_admin_invite_create()] {
            input[type="submit", value="Generate new invite code"];
        }
        ul { @for t in *invites {
            li {
                form[method="POST", action=u_admin_invite_remove()] {
                    span { @t }
                    input[type="text", name="invite", value=&t, hidden];
                    input[type="submit", value="Invalidate"];
                }
            }
        }}

        // h2 { "Database" }
        // @match db_stats(&database) {
        //     Ok(s) => { @s }
        //     Err(e) => { pre.error { @format!("{e:?}") } }
        // }
    }
);