Compare commits
6 Commits
4a06d519bf
...
2e3be062c6
Author | SHA1 | Date | |
---|---|---|---|
2e3be062c6 | |||
6d61d1f8bc | |||
b7499bd6cb | |||
60b9a4dbba | |||
a1a5e2ad89 | |||
eda072c713 |
@ -15,7 +15,6 @@ test('cox can create and delete trip', async ({ page }) => {
|
|||||||
await page.locator('#sidebar #planned_starting_time').press('Tab');
|
await page.locator('#sidebar #planned_starting_time').press('Tab');
|
||||||
await page.getByRole('spinbutton').fill('5');
|
await page.getByRole('spinbutton').fill('5');
|
||||||
await page.getByRole('button', { name: 'Erstellen', exact: true }).click();
|
await page.getByRole('button', { name: 'Erstellen', exact: true }).click();
|
||||||
await page.getByRole('link', { name: 'Geplante Ausfahrten' }).click();
|
|
||||||
await expect(page.locator('body')).toContainText('18:00 Uhr (cox) Details');
|
await expect(page.locator('body')).toContainText('18:00 Uhr (cox) Details');
|
||||||
|
|
||||||
await page.goto('http://localhost:8000/planned');
|
await page.goto('http://localhost:8000/planned');
|
||||||
@ -57,7 +56,6 @@ test.describe('cox can edit trips', () => {
|
|||||||
await sharedPage.locator('#sidebar #notes').click();
|
await sharedPage.locator('#sidebar #notes').click();
|
||||||
await sharedPage.locator('#sidebar #notes').fill('Meine Anmerkung');
|
await sharedPage.locator('#sidebar #notes').fill('Meine Anmerkung');
|
||||||
await sharedPage.getByRole('button', { name: 'Speichern' }).click();
|
await sharedPage.getByRole('button', { name: 'Speichern' }).click();
|
||||||
await sharedPage.getByRole('link', { name: 'Geplante Ausfahrten' }).click();
|
|
||||||
await sharedPage.getByRole('link', { name: 'Details' }).click();
|
await sharedPage.getByRole('link', { name: 'Details' }).click();
|
||||||
await expect(sharedPage.locator('#sidebar')).toContainText('Meine Anmerkung');
|
await expect(sharedPage.locator('#sidebar')).toContainText('Meine Anmerkung');
|
||||||
|
|
||||||
@ -94,7 +92,6 @@ test.describe('cox can edit trips', () => {
|
|||||||
await sharedPage.getByRole('spinbutton').fill('3');
|
await sharedPage.getByRole('spinbutton').fill('3');
|
||||||
await sharedPage.getByRole('button', { name: 'Speichern' }).click();
|
await sharedPage.getByRole('button', { name: 'Speichern' }).click();
|
||||||
await expect(sharedPage.locator('body')).toContainText('Ausfahrt erfolgreich aktualisiert.');
|
await expect(sharedPage.locator('body')).toContainText('Ausfahrt erfolgreich aktualisiert.');
|
||||||
await sharedPage.getByRole('link', { name: 'Geplante Ausfahrten' }).click();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test('call off trip', async () => {
|
test('call off trip', async () => {
|
||||||
@ -105,7 +102,6 @@ test.describe('cox can edit trips', () => {
|
|||||||
await sharedPage.getByRole('spinbutton').fill('0');
|
await sharedPage.getByRole('spinbutton').fill('0');
|
||||||
await sharedPage.getByRole('button', { name: 'Speichern' }).click();
|
await sharedPage.getByRole('button', { name: 'Speichern' }).click();
|
||||||
await expect(sharedPage.locator('body')).toContainText('Ausfahrt erfolgreich aktualisiert.');
|
await expect(sharedPage.locator('body')).toContainText('Ausfahrt erfolgreich aktualisiert.');
|
||||||
await sharedPage.getByRole('link', { name: 'Geplante Ausfahrten' }).click();
|
|
||||||
await expect(sharedPage.locator('body')).toContainText('(Absage cox )');
|
await expect(sharedPage.locator('body')).toContainText('(Absage cox )');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -3,10 +3,12 @@ INSERT INTO "role" (name) VALUES ('cox');
|
|||||||
INSERT INTO "role" (name) VALUES ('scheckbuch');
|
INSERT INTO "role" (name) VALUES ('scheckbuch');
|
||||||
INSERT INTO "role" (name) VALUES ('tech');
|
INSERT INTO "role" (name) VALUES ('tech');
|
||||||
INSERT INTO "role" (name) VALUES ('Donau Linz');
|
INSERT INTO "role" (name) VALUES ('Donau Linz');
|
||||||
|
INSERT INTO "role" (name) VALUES ('planned_event');
|
||||||
INSERT INTO "user" (name, pw) VALUES('admin', '$argon2id$v=19$m=19456,t=2,p=1$dS/X5/sPEKTj4Rzs/CuvzQ$4P4NCw4Ukhv80/eQYTsarHhnw61JuL1KMx/L9dm82YM');
|
INSERT INTO "user" (name, pw) VALUES('admin', '$argon2id$v=19$m=19456,t=2,p=1$dS/X5/sPEKTj4Rzs/CuvzQ$4P4NCw4Ukhv80/eQYTsarHhnw61JuL1KMx/L9dm82YM');
|
||||||
INSERT INTO "user_role" (user_id, role_id) VALUES(1,1);
|
INSERT INTO "user_role" (user_id, role_id) VALUES(1,1);
|
||||||
INSERT INTO "user_role" (user_id, role_id) VALUES(1,2);
|
INSERT INTO "user_role" (user_id, role_id) VALUES(1,2);
|
||||||
INSERT INTO "user_role" (user_id, role_id) VALUES(1,5);
|
INSERT INTO "user_role" (user_id, role_id) VALUES(1,5);
|
||||||
|
INSERT INTO "user_role" (user_id, role_id) VALUES(1,6);
|
||||||
INSERT INTO "user" (name, pw) VALUES('rower', '$argon2id$v=19$m=19456,t=2,p=1$dS/X5/sPEKTj4Rzs/CuvzQ$jWKzDmI0jqT2dqINFt6/1NjVF4Dx15n07PL1ZMBmFsY');
|
INSERT INTO "user" (name, pw) VALUES('rower', '$argon2id$v=19$m=19456,t=2,p=1$dS/X5/sPEKTj4Rzs/CuvzQ$jWKzDmI0jqT2dqINFt6/1NjVF4Dx15n07PL1ZMBmFsY');
|
||||||
INSERT INTO "user_role" (user_id, role_id) VALUES(2,5);
|
INSERT INTO "user_role" (user_id, role_id) VALUES(2,5);
|
||||||
INSERT INTO "user" (name, pw) VALUES('guest', '$argon2id$v=19$m=19456,t=2,p=1$dS/X5/sPEKTj4Rzs/CuvzQ$GF6gizbI79Bh0zA9its8S0gram956v+YIV8w8VpwJnQ');
|
INSERT INTO "user" (name, pw) VALUES('guest', '$argon2id$v=19$m=19456,t=2,p=1$dS/X5/sPEKTj4Rzs/CuvzQ$GF6gizbI79Bh0zA9its8S0gram956v+YIV8w8VpwJnQ');
|
||||||
|
@ -1,12 +1,8 @@
|
|||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
|
|
||||||
use lettre::{
|
use lettre::{
|
||||||
message::{
|
message::header::ContentType, transport::smtp::authentication::Credentials, Message,
|
||||||
header::{self, ContentType},
|
SmtpTransport, Transport,
|
||||||
MultiPart, SinglePart,
|
|
||||||
},
|
|
||||||
transport::smtp::authentication::Credentials,
|
|
||||||
Message, SmtpTransport, Transport,
|
|
||||||
};
|
};
|
||||||
use sqlx::SqlitePool;
|
use sqlx::SqlitePool;
|
||||||
|
|
||||||
@ -17,7 +13,7 @@ use super::{family::Family, role::Role, user::User};
|
|||||||
pub struct Mail {}
|
pub struct Mail {}
|
||||||
|
|
||||||
impl Mail {
|
impl Mail {
|
||||||
pub async fn send(db: &SqlitePool, data: MailToSend<'_>, smtp_pw: String) -> bool {
|
pub async fn send(db: &SqlitePool, data: MailToSend, smtp_pw: String) -> bool {
|
||||||
let mut email = Message::builder()
|
let mut email = Message::builder()
|
||||||
.from(
|
.from(
|
||||||
"ASKÖ Ruderverein Donau Linz <no-reply@rudernlinz.at>"
|
"ASKÖ Ruderverein Donau Linz <no-reply@rudernlinz.at>"
|
||||||
|
@ -3,8 +3,8 @@ use sqlx::{FromRow, SqlitePool};
|
|||||||
|
|
||||||
#[derive(FromRow, Serialize, Clone)]
|
#[derive(FromRow, Serialize, Clone)]
|
||||||
pub struct Role {
|
pub struct Role {
|
||||||
id: i64,
|
pub(crate) id: i64,
|
||||||
name: String,
|
pub(crate) name: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Role {
|
impl Role {
|
||||||
@ -30,6 +30,21 @@ WHERE id like ?
|
|||||||
.ok()
|
.ok()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn find_by_name(db: &SqlitePool, name: &str) -> Option<Self> {
|
||||||
|
sqlx::query_as!(
|
||||||
|
Self,
|
||||||
|
"
|
||||||
|
SELECT id, name
|
||||||
|
FROM role
|
||||||
|
WHERE name like ?
|
||||||
|
",
|
||||||
|
name
|
||||||
|
)
|
||||||
|
.fetch_one(db)
|
||||||
|
.await
|
||||||
|
.ok()
|
||||||
|
}
|
||||||
|
|
||||||
pub async fn mails_from_role(&self, db: &SqlitePool) -> Vec<String> {
|
pub async fn mails_from_role(&self, db: &SqlitePool) -> Vec<String> {
|
||||||
let query = format!(
|
let query = format!(
|
||||||
"SELECT u.mail
|
"SELECT u.mail
|
||||||
|
@ -2,7 +2,6 @@ use std::ops::{Deref, DerefMut};
|
|||||||
|
|
||||||
use argon2::{password_hash::SaltString, Argon2, PasswordHasher};
|
use argon2::{password_hash::SaltString, Argon2, PasswordHasher};
|
||||||
use chrono::{Datelike, Local, NaiveDate};
|
use chrono::{Datelike, Local, NaiveDate};
|
||||||
use chrono_tz::Etc::UTC;
|
|
||||||
use log::info;
|
use log::info;
|
||||||
use rocket::{
|
use rocket::{
|
||||||
async_trait,
|
async_trait,
|
||||||
@ -14,7 +13,7 @@ use rocket::{
|
|||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use sqlx::{FromRow, Sqlite, SqlitePool, Transaction};
|
use sqlx::{FromRow, Sqlite, SqlitePool, Transaction};
|
||||||
|
|
||||||
use super::{family::Family, log::Log, tripdetails::TripDetails, Day};
|
use super::{family::Family, log::Log, role::Role, tripdetails::TripDetails, Day};
|
||||||
use crate::tera::admin::user::UserEditForm;
|
use crate::tera::admin::user::UserEditForm;
|
||||||
|
|
||||||
const RENNRUDERBEITRAG: i32 = 11000;
|
const RENNRUDERBEITRAG: i32 = 11000;
|
||||||
@ -101,10 +100,12 @@ pub enum LoginError {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Serialize)]
|
#[derive(Debug, Serialize)]
|
||||||
pub(crate) struct Fee {
|
pub struct Fee {
|
||||||
pub(crate) sum_in_cents: i32,
|
pub sum_in_cents: i32,
|
||||||
pub(crate) parts: Vec<(String, i32)>,
|
pub parts: Vec<(String, i32)>,
|
||||||
pub(crate) name: String,
|
pub name: String,
|
||||||
|
pub user_ids: String,
|
||||||
|
pub paid: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Fee {
|
impl Fee {
|
||||||
@ -113,6 +114,8 @@ impl Fee {
|
|||||||
sum_in_cents: 0,
|
sum_in_cents: 0,
|
||||||
name: "".into(),
|
name: "".into(),
|
||||||
parts: Vec::new(),
|
parts: Vec::new(),
|
||||||
|
user_ids: "".into(),
|
||||||
|
paid: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -122,8 +125,18 @@ impl Fee {
|
|||||||
self.parts.push((desc, price_in_cents));
|
self.parts.push((desc, price_in_cents));
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn name(&mut self, name: String) {
|
pub fn add_person(&mut self, user: &User) {
|
||||||
self.name = name;
|
if !self.name.is_empty() {
|
||||||
|
self.name.push_str(" + ");
|
||||||
|
self.user_ids.push_str("&");
|
||||||
|
}
|
||||||
|
self.name.push_str(&user.name);
|
||||||
|
|
||||||
|
self.user_ids.push_str(&format!("user_ids[]={}", user.id));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn paid(&mut self) {
|
||||||
|
self.paid = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn merge(&mut self, fee: Fee) {
|
pub fn merge(&mut self, fee: Fee) {
|
||||||
@ -142,22 +155,23 @@ impl User {
|
|||||||
let mut fee = Fee::new();
|
let mut fee = Fee::new();
|
||||||
|
|
||||||
if let Some(family) = Family::find_by_opt_id(db, self.family_id).await {
|
if let Some(family) = Family::find_by_opt_id(db, self.family_id).await {
|
||||||
let mut names = String::new();
|
|
||||||
for member in family.members(db).await {
|
for member in family.members(db).await {
|
||||||
if !names.is_empty() {
|
fee.add_person(&member);
|
||||||
names.push_str(" + ");
|
if member.has_role(db, "paid").await {
|
||||||
|
fee.paid();
|
||||||
}
|
}
|
||||||
names.push_str(&member.name);
|
|
||||||
fee.merge(member.fee_without_families(db).await);
|
fee.merge(member.fee_without_families(db).await);
|
||||||
}
|
}
|
||||||
fee.name(names);
|
|
||||||
if family.amount_family_members(db).await > 2 {
|
if family.amount_family_members(db).await > 2 {
|
||||||
fee.add("Familie 3+ Personen".into(), FAMILY_THREE_OR_MORE);
|
fee.add("Familie 3+ Personen".into(), FAMILY_THREE_OR_MORE);
|
||||||
} else {
|
} else {
|
||||||
fee.add("Familie 2 Personen".into(), FAMILY_TWO);
|
fee.add("Familie 2 Personen".into(), FAMILY_TWO);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
fee.name(self.name.clone());
|
fee.add_person(&self);
|
||||||
|
if self.has_role(db, "paid").await {
|
||||||
|
fee.paid();
|
||||||
|
}
|
||||||
fee.merge(self.fee_without_families(db).await);
|
fee.merge(self.fee_without_families(db).await);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -452,15 +466,36 @@ ORDER BY last_access DESC
|
|||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
for role_id in data.roles.into_keys() {
|
for role_id in data.roles.into_keys() {
|
||||||
|
self.add_role(
|
||||||
|
db,
|
||||||
|
&Role::find_by_id(db, role_id.parse::<i32>().unwrap())
|
||||||
|
.await
|
||||||
|
.unwrap(),
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn add_role(&self, db: &SqlitePool, role: &Role) {
|
||||||
sqlx::query!(
|
sqlx::query!(
|
||||||
"INSERT INTO user_role(user_id, role_id) VALUES (?, ?)",
|
"INSERT INTO user_role(user_id, role_id) VALUES (?, ?)",
|
||||||
self.id,
|
self.id,
|
||||||
role_id
|
role.id
|
||||||
)
|
)
|
||||||
.execute(db)
|
.execute(db)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn remove_role(&self, db: &SqlitePool, role: &Role) {
|
||||||
|
sqlx::query!(
|
||||||
|
"DELETE FROM user_role WHERE user_id = ? and role_id = ?",
|
||||||
|
self.id,
|
||||||
|
role.id
|
||||||
|
)
|
||||||
|
.execute(db)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn login(db: &SqlitePool, name: &str, pw: &str) -> Result<Self, LoginError> {
|
pub async fn login(db: &SqlitePool, name: &str, pw: &str) -> Result<Self, LoginError> {
|
||||||
@ -813,6 +848,43 @@ impl<'r> FromRequest<'r> for VorstandUser {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
|
pub struct PlannedEventUser(pub(crate) User);
|
||||||
|
|
||||||
|
impl Into<User> for PlannedEventUser {
|
||||||
|
fn into(self) -> User {
|
||||||
|
self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deref for PlannedEventUser {
|
||||||
|
type Target = User;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
&self.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl<'r> FromRequest<'r> for PlannedEventUser {
|
||||||
|
type Error = LoginError;
|
||||||
|
|
||||||
|
async fn from_request(req: &'r Request<'_>) -> request::Outcome<Self, Self::Error> {
|
||||||
|
let db = req.rocket().state::<SqlitePool>().unwrap();
|
||||||
|
match User::from_request(req).await {
|
||||||
|
Outcome::Success(user) => {
|
||||||
|
if user.has_role(db, "planned_event").await {
|
||||||
|
Outcome::Success(PlannedEventUser(user))
|
||||||
|
} else {
|
||||||
|
Outcome::Error((Status::Forbidden, LoginError::NotACox))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Outcome::Error(f) => Outcome::Error(f),
|
||||||
|
Outcome::Forward(f) => Outcome::Forward(f),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use rocket::{form::Form, fs::FileServer, post, routes, Build, FromForm, Rocket, State};
|
use rocket::{form::Form, post, routes, Build, FromForm, Rocket, State};
|
||||||
use serde_json::json;
|
use serde_json::json;
|
||||||
use sqlx::SqlitePool;
|
use sqlx::SqlitePool;
|
||||||
|
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
use rocket::form::Form;
|
use rocket::form::Form;
|
||||||
use rocket::fs::TempFile;
|
|
||||||
use rocket::response::{Flash, Redirect};
|
use rocket::response::{Flash, Redirect};
|
||||||
use rocket::{get, request::FlashMessage, routes, Route, State};
|
use rocket::{get, request::FlashMessage, routes, Route, State};
|
||||||
use rocket::{post, FromForm};
|
use rocket::{post, FromForm};
|
||||||
@ -34,28 +33,24 @@ async fn index(
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[get("/mail/fee")]
|
#[get("/mail/fee")]
|
||||||
async fn fee(
|
async fn fee(db: &State<SqlitePool>, _admin: AdminUser, config: &State<Config>) -> &'static str {
|
||||||
db: &State<SqlitePool>,
|
|
||||||
admin: AdminUser,
|
|
||||||
config: &State<Config>,
|
|
||||||
flash: Option<FlashMessage<'_>>,
|
|
||||||
) -> &'static str {
|
|
||||||
Mail::fees(db, config.smtp_pw.clone()).await;
|
Mail::fees(db, config.smtp_pw.clone()).await;
|
||||||
"SUCC"
|
"SUCC"
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(FromForm, Debug)]
|
#[derive(FromForm, Debug)]
|
||||||
pub struct MailToSend<'a> {
|
pub struct MailToSend {
|
||||||
|
//<'a> {
|
||||||
pub(crate) role_id: i32,
|
pub(crate) role_id: i32,
|
||||||
pub(crate) subject: String,
|
pub(crate) subject: String,
|
||||||
pub(crate) body: String,
|
pub(crate) body: String,
|
||||||
pub(crate) files: Vec<TempFile<'a>>,
|
//pub(crate) files: Vec<TempFile<'a>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[post("/mail", data = "<data>")]
|
#[post("/mail", data = "<data>")]
|
||||||
async fn update(
|
async fn update(
|
||||||
db: &State<SqlitePool>,
|
db: &State<SqlitePool>,
|
||||||
data: Form<MailToSend<'_>>,
|
data: Form<MailToSend>,
|
||||||
config: &State<Config>,
|
config: &State<Config>,
|
||||||
_admin: AdminUser,
|
_admin: AdminUser,
|
||||||
) -> Flash<Redirect> {
|
) -> Flash<Redirect> {
|
||||||
|
@ -10,7 +10,7 @@ use sqlx::SqlitePool;
|
|||||||
use crate::model::{
|
use crate::model::{
|
||||||
planned_event::PlannedEvent,
|
planned_event::PlannedEvent,
|
||||||
tripdetails::{TripDetails, TripDetailsToAdd},
|
tripdetails::{TripDetails, TripDetailsToAdd},
|
||||||
user::AdminUser,
|
user::PlannedEventUser,
|
||||||
};
|
};
|
||||||
|
|
||||||
//TODO: add constraints (e.g. planned_amount_cox > 0)
|
//TODO: add constraints (e.g. planned_amount_cox > 0)
|
||||||
@ -25,7 +25,7 @@ struct AddPlannedEventForm<'r> {
|
|||||||
async fn create(
|
async fn create(
|
||||||
db: &State<SqlitePool>,
|
db: &State<SqlitePool>,
|
||||||
data: Form<AddPlannedEventForm<'_>>,
|
data: Form<AddPlannedEventForm<'_>>,
|
||||||
_admin: AdminUser,
|
_admin: PlannedEventUser,
|
||||||
) -> Flash<Redirect> {
|
) -> Flash<Redirect> {
|
||||||
let data = data.into_inner();
|
let data = data.into_inner();
|
||||||
|
|
||||||
@ -36,7 +36,7 @@ async fn create(
|
|||||||
|
|
||||||
PlannedEvent::create(db, data.name, data.planned_amount_cox, trip_details).await;
|
PlannedEvent::create(db, data.name, data.planned_amount_cox, trip_details).await;
|
||||||
|
|
||||||
Flash::success(Redirect::to("/"), "Event hinzugefügt")
|
Flash::success(Redirect::to("/planned"), "Event hinzugefügt")
|
||||||
}
|
}
|
||||||
|
|
||||||
//TODO: add constraints (e.g. planned_amount_cox > 0)
|
//TODO: add constraints (e.g. planned_amount_cox > 0)
|
||||||
@ -54,7 +54,7 @@ struct UpdatePlannedEventForm<'r> {
|
|||||||
async fn update(
|
async fn update(
|
||||||
db: &State<SqlitePool>,
|
db: &State<SqlitePool>,
|
||||||
data: Form<UpdatePlannedEventForm<'_>>,
|
data: Form<UpdatePlannedEventForm<'_>>,
|
||||||
_admin: AdminUser,
|
_admin: PlannedEventUser,
|
||||||
) -> Flash<Redirect> {
|
) -> Flash<Redirect> {
|
||||||
match PlannedEvent::find_by_id(db, data.id).await {
|
match PlannedEvent::find_by_id(db, data.id).await {
|
||||||
Some(planned_event) => {
|
Some(planned_event) => {
|
||||||
@ -68,20 +68,20 @@ async fn update(
|
|||||||
data.is_locked,
|
data.is_locked,
|
||||||
)
|
)
|
||||||
.await;
|
.await;
|
||||||
Flash::success(Redirect::to("/"), "Successfully edited the event")
|
Flash::success(Redirect::to("/planned"), "Event erfolgreich bearbeitet")
|
||||||
}
|
}
|
||||||
None => Flash::error(Redirect::to("/"), "Planned event id not found"),
|
None => Flash::error(Redirect::to("/planned"), "Planned event id not found"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[get("/planned-event/<id>/delete")]
|
#[get("/planned-event/<id>/delete")]
|
||||||
async fn delete(db: &State<SqlitePool>, id: i64, _admin: AdminUser) -> Flash<Redirect> {
|
async fn delete(db: &State<SqlitePool>, id: i64, _admin: PlannedEventUser) -> Flash<Redirect> {
|
||||||
match PlannedEvent::find_by_id(db, id).await {
|
match PlannedEvent::find_by_id(db, id).await {
|
||||||
Some(planned_event) => {
|
Some(planned_event) => {
|
||||||
planned_event.delete(db).await;
|
planned_event.delete(db).await;
|
||||||
Flash::success(Redirect::to("/"), "Event gelöscht")
|
Flash::success(Redirect::to("/planned"), "Event gelöscht")
|
||||||
}
|
}
|
||||||
None => Flash::error(Redirect::to("/"), "PlannedEvent does not exist"),
|
None => Flash::error(Redirect::to("/planned"), "PlannedEvent does not exist"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -120,7 +120,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -151,7 +151,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -187,7 +187,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -196,7 +196,7 @@ mod test {
|
|||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
flash_cookie.value(),
|
flash_cookie.value(),
|
||||||
"7:successSuccessfully edited the event"
|
"7:successEvent erfolgreich bearbeitet"
|
||||||
);
|
);
|
||||||
|
|
||||||
let event = PlannedEvent::find_by_id(&db, 1).await.unwrap();
|
let event = PlannedEvent::find_by_id(&db, 1).await.unwrap();
|
||||||
@ -224,7 +224,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -255,7 +255,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
|
@ -3,9 +3,9 @@ use std::collections::HashMap;
|
|||||||
use crate::model::{
|
use crate::model::{
|
||||||
family::Family,
|
family::Family,
|
||||||
role::Role,
|
role::Role,
|
||||||
user::{AdminUser, Fee, User, UserWithRoles, VorstandUser},
|
user::{AdminUser, User, UserWithRoles, VorstandUser},
|
||||||
};
|
};
|
||||||
use futures::future::{self, join_all};
|
use futures::future::join_all;
|
||||||
use rocket::{
|
use rocket::{
|
||||||
form::Form,
|
form::Form,
|
||||||
get, post,
|
get, post,
|
||||||
@ -77,6 +77,33 @@ async fn fees(
|
|||||||
Template::render("admin/user/fees", context.into_json())
|
Template::render("admin/user/fees", context.into_json())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[get("/user/fees/paid?<user_ids>")]
|
||||||
|
async fn fees_paid(
|
||||||
|
db: &State<SqlitePool>,
|
||||||
|
_admin: AdminUser,
|
||||||
|
user_ids: Vec<i32>,
|
||||||
|
) -> Flash<Redirect> {
|
||||||
|
let mut res = String::new();
|
||||||
|
for user_id in user_ids {
|
||||||
|
let user = User::find_by_id(db, user_id).await.unwrap();
|
||||||
|
res.push_str(&format!("{} + ", user.name));
|
||||||
|
if user.has_role(db, "paid").await {
|
||||||
|
user.remove_role(db, &Role::find_by_name(db, "paid").await.unwrap())
|
||||||
|
.await;
|
||||||
|
} else {
|
||||||
|
user.add_role(db, &Role::find_by_name(db, "paid").await.unwrap())
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
res.truncate(res.len() - 3); // remove ' + ' from the end
|
||||||
|
|
||||||
|
Flash::success(
|
||||||
|
Redirect::to("/admin/user/fees"),
|
||||||
|
format!("Zahlungsstatus von {} erfolgreich geändert", res),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
#[get("/user/<user>/reset-pw")]
|
#[get("/user/<user>/reset-pw")]
|
||||||
async fn resetpw(db: &State<SqlitePool>, _admin: AdminUser, user: i32) -> Flash<Redirect> {
|
async fn resetpw(db: &State<SqlitePool>, _admin: AdminUser, user: i32) -> Flash<Redirect> {
|
||||||
let user = User::find_by_id(db, user).await;
|
let user = User::find_by_id(db, user).await;
|
||||||
@ -165,5 +192,5 @@ async fn create(
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn routes() -> Vec<Route> {
|
pub fn routes() -> Vec<Route> {
|
||||||
routes![index, resetpw, update, create, delete, fees]
|
routes![index, resetpw, update, create, delete, fees, fees_paid]
|
||||||
}
|
}
|
||||||
|
@ -89,7 +89,15 @@ async fn login(
|
|||||||
)
|
)
|
||||||
.await;
|
.await;
|
||||||
|
|
||||||
Flash::success(Redirect::to("/"), "Login erfolgreich")
|
// Check for redirect_url cookie and redirect accordingly
|
||||||
|
match cookies.get_private("redirect_url") {
|
||||||
|
Some(redirect_cookie) => {
|
||||||
|
let redirect_url = redirect_cookie.value().to_string();
|
||||||
|
cookies.remove_private(redirect_cookie); // Remove the cookie after using it
|
||||||
|
Flash::success(Redirect::to(redirect_url), "Login erfolgreich")
|
||||||
|
}
|
||||||
|
None => Flash::success(Redirect::to("/"), "Login erfolgreich"),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[get("/set-pw/<userid>")]
|
#[get("/set-pw/<userid>")]
|
||||||
|
@ -34,7 +34,7 @@ async fn create(
|
|||||||
//)
|
//)
|
||||||
//.await;
|
//.await;
|
||||||
|
|
||||||
Flash::success(Redirect::to("/"), "Ausfahrt erfolgreich erstellt.")
|
Flash::success(Redirect::to("/planned"), "Ausfahrt erfolgreich erstellt.")
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(FromForm)]
|
#[derive(FromForm)]
|
||||||
@ -66,16 +66,19 @@ async fn update(
|
|||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
{
|
{
|
||||||
Ok(_) => Flash::success(Redirect::to("/"), "Ausfahrt erfolgreich aktualisiert."),
|
Ok(_) => Flash::success(
|
||||||
|
Redirect::to("/planned"),
|
||||||
|
"Ausfahrt erfolgreich aktualisiert.",
|
||||||
|
),
|
||||||
Err(TripUpdateError::NotYourTrip) => {
|
Err(TripUpdateError::NotYourTrip) => {
|
||||||
Flash::error(Redirect::to("/"), "Nicht deine Ausfahrt!")
|
Flash::error(Redirect::to("/planned"), "Nicht deine Ausfahrt!")
|
||||||
}
|
}
|
||||||
Err(TripUpdateError::TripDetailsDoesNotExist) => {
|
Err(TripUpdateError::TripDetailsDoesNotExist) => {
|
||||||
Flash::error(Redirect::to("/"), "Ausfahrt gibt's nicht")
|
Flash::error(Redirect::to("/planned"), "Ausfahrt gibt's nicht")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Flash::error(Redirect::to("/"), "Ausfahrt gibt's nicht")
|
Flash::error(Redirect::to("/planned"), "Ausfahrt gibt's nicht")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -92,21 +95,21 @@ async fn join(db: &State<SqlitePool>, planned_event_id: i64, cox: CoxUser) -> Fl
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
.await;
|
.await;
|
||||||
Flash::success(Redirect::to("/"), "Danke für's helfen!")
|
Flash::success(Redirect::to("/planned"), "Danke für's helfen!")
|
||||||
}
|
}
|
||||||
Err(CoxHelpError::AlreadyRegisteredAsCox) => {
|
Err(CoxHelpError::AlreadyRegisteredAsCox) => {
|
||||||
Flash::error(Redirect::to("/"), "Du hilfst bereits aus!")
|
Flash::error(Redirect::to("/planned"), "Du hilfst bereits aus!")
|
||||||
}
|
}
|
||||||
Err(CoxHelpError::AlreadyRegisteredAsRower) => Flash::error(
|
Err(CoxHelpError::AlreadyRegisteredAsRower) => Flash::error(
|
||||||
Redirect::to("/"),
|
Redirect::to("/planned"),
|
||||||
"Du hast dich bereits als Ruderer angemeldet!",
|
"Du hast dich bereits als Ruderer angemeldet!",
|
||||||
),
|
),
|
||||||
Err(CoxHelpError::DetailsLocked) => {
|
Err(CoxHelpError::DetailsLocked) => {
|
||||||
Flash::error(Redirect::to("/"), "Boot ist bereits eingeteilt.")
|
Flash::error(Redirect::to("/planned"), "Boot ist bereits eingeteilt.")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Flash::error(Redirect::to("/"), "Event gibt's nicht")
|
Flash::error(Redirect::to("/planned"), "Event gibt's nicht")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -114,18 +117,18 @@ async fn join(db: &State<SqlitePool>, planned_event_id: i64, cox: CoxUser) -> Fl
|
|||||||
async fn remove_trip(db: &State<SqlitePool>, trip_id: i64, cox: CoxUser) -> Flash<Redirect> {
|
async fn remove_trip(db: &State<SqlitePool>, trip_id: i64, cox: CoxUser) -> Flash<Redirect> {
|
||||||
let trip = Trip::find_by_id(db, trip_id).await;
|
let trip = Trip::find_by_id(db, trip_id).await;
|
||||||
match trip {
|
match trip {
|
||||||
None => Flash::error(Redirect::to("/"), "Trip gibt's nicht!"),
|
None => Flash::error(Redirect::to("/planned"), "Trip gibt's nicht!"),
|
||||||
Some(trip) => match trip.delete(db, &cox).await {
|
Some(trip) => match trip.delete(db, &cox).await {
|
||||||
Ok(_) => {
|
Ok(_) => {
|
||||||
Log::create(db, format!("Cox {} deleted trip.id={}", cox.name, trip_id)).await;
|
Log::create(db, format!("Cox {} deleted trip.id={}", cox.name, trip_id)).await;
|
||||||
Flash::success(Redirect::to("/"), "Erfolgreich gelöscht!")
|
Flash::success(Redirect::to("/planned"), "Erfolgreich gelöscht!")
|
||||||
}
|
}
|
||||||
Err(TripDeleteError::SomebodyAlreadyRegistered) => Flash::error(
|
Err(TripDeleteError::SomebodyAlreadyRegistered) => Flash::error(
|
||||||
Redirect::to("/"),
|
Redirect::to("/planned"),
|
||||||
"Ausfahrt kann nicht gelöscht werden, da bereits jemand registriert ist!",
|
"Ausfahrt kann nicht gelöscht werden, da bereits jemand registriert ist!",
|
||||||
),
|
),
|
||||||
Err(TripDeleteError::NotYourTrip) => {
|
Err(TripDeleteError::NotYourTrip) => {
|
||||||
Flash::error(Redirect::to("/"), "Nicht deine Ausfahrt!")
|
Flash::error(Redirect::to("/planned"), "Nicht deine Ausfahrt!")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@ -145,17 +148,17 @@ async fn remove(db: &State<SqlitePool>, planned_event_id: i64, cox: CoxUser) ->
|
|||||||
)
|
)
|
||||||
.await;
|
.await;
|
||||||
|
|
||||||
Flash::success(Redirect::to("/"), "Erfolgreich abgemeldet!")
|
Flash::success(Redirect::to("/planned"), "Erfolgreich abgemeldet!")
|
||||||
}
|
}
|
||||||
Err(TripHelpDeleteError::DetailsLocked) => {
|
Err(TripHelpDeleteError::DetailsLocked) => {
|
||||||
Flash::error(Redirect::to("/"), "Boot bereits eingeteilt")
|
Flash::error(Redirect::to("/planned"), "Boot bereits eingeteilt")
|
||||||
}
|
}
|
||||||
Err(TripHelpDeleteError::CoxNotHelping) => {
|
Err(TripHelpDeleteError::CoxNotHelping) => {
|
||||||
Flash::error(Redirect::to("/"), "Steuermann hilft nicht aus...")
|
Flash::error(Redirect::to("/planned"), "Steuermann hilft nicht aus...")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Flash::error(Redirect::to("/"), "Planned_event does not exist.")
|
Flash::error(Redirect::to("/planned"), "Planned_event does not exist.")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -202,7 +205,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -250,7 +253,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -288,7 +291,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -326,7 +329,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -354,7 +357,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -367,7 +370,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -398,7 +401,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -429,7 +432,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -470,7 +473,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -498,7 +501,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
@ -526,7 +529,7 @@ mod test {
|
|||||||
let response = req.dispatch().await;
|
let response = req.dispatch().await;
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::SeeOther);
|
assert_eq!(response.status(), Status::SeeOther);
|
||||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
assert_eq!(response.headers().get("Location").next(), Some("/planned"));
|
||||||
|
|
||||||
let flash_cookie = response
|
let flash_cookie = response
|
||||||
.cookies()
|
.cookies()
|
||||||
|
@ -125,7 +125,7 @@ async fn new_kiosk(
|
|||||||
async fn kiosk(
|
async fn kiosk(
|
||||||
db: &State<SqlitePool>,
|
db: &State<SqlitePool>,
|
||||||
flash: Option<FlashMessage<'_>>,
|
flash: Option<FlashMessage<'_>>,
|
||||||
kiosk: KioskCookie,
|
_kiosk: KioskCookie,
|
||||||
) -> Template {
|
) -> Template {
|
||||||
let boats = Boat::all(db).await;
|
let boats = Boat::all(db).await;
|
||||||
let coxes: Vec<UserWithWaterStatus> = futures::future::join_all(
|
let coxes: Vec<UserWithWaterStatus> = futures::future::join_all(
|
||||||
|
@ -3,10 +3,14 @@ use rocket::{
|
|||||||
fairing::AdHoc,
|
fairing::AdHoc,
|
||||||
form::Form,
|
form::Form,
|
||||||
fs::FileServer,
|
fs::FileServer,
|
||||||
get, post,
|
get,
|
||||||
|
http::Cookie,
|
||||||
|
post,
|
||||||
request::FlashMessage,
|
request::FlashMessage,
|
||||||
response::{Flash, Redirect},
|
response::{Flash, Redirect},
|
||||||
routes, Build, FromForm, Rocket, State,
|
routes,
|
||||||
|
time::{Duration, OffsetDateTime},
|
||||||
|
Build, FromForm, Request, Rocket, State,
|
||||||
};
|
};
|
||||||
use rocket_dyn_templates::Template;
|
use rocket_dyn_templates::Template;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
@ -51,7 +55,13 @@ async fn wikiauth(db: &State<SqlitePool>, login: Form<LoginForm<'_>>) -> String
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[catch(401)] //Unauthorized
|
#[catch(401)] //Unauthorized
|
||||||
fn unauthorized_error() -> Redirect {
|
fn unauthorized_error(req: &Request) -> Redirect {
|
||||||
|
// Save the URL the user tried to access, to be able to go there once logged in
|
||||||
|
let mut redirect_cookie = Cookie::new("redirect_url", format!("{}", req.uri()));
|
||||||
|
println!("{}", req.uri());
|
||||||
|
redirect_cookie.set_expires(OffsetDateTime::now_utc() + Duration::hours(1));
|
||||||
|
req.cookies().add_private(redirect_cookie);
|
||||||
|
|
||||||
Redirect::to("/auth")
|
Redirect::to("/auth")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,29 +3,45 @@
|
|||||||
{% extends "base" %}
|
{% extends "base" %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="max-w-screen-lg w-full">
|
<div class="max-w-screen-lg w-full bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5">
|
||||||
|
|
||||||
{% if flash %}
|
{% if flash %}
|
||||||
{{ macros::alert(message=flash.1, type=flash.0, class="my-3") }}
|
{{ macros::alert(message=flash.1, type=flash.0, class="sm:col-span-2 lg:col-span-3") }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
<div class="grid gap-3">
|
<h1 class="h1">Gebühren</h1>
|
||||||
<div class="bg-white dark:bg-primary-900 text-black dark:text-white rounded-md block shadow mt-5" role="alert">
|
|
||||||
<h2 class="h2">Gebühren</h2>
|
<!-- START filterBar -->
|
||||||
<div class="text-sm p-3">
|
<div class="search-wrapper">
|
||||||
|
<label for="name" class="sr-only">Suche</label>
|
||||||
|
<input type="search" name="name" id="filter-js" class="search-bar" placeholder="Suchen nach Name"/>
|
||||||
|
</div>
|
||||||
|
<!-- END filterBar -->
|
||||||
|
|
||||||
|
<div class="bg-primary-100 dark:bg-primary-950 p-3 rounded-b-md grid gap-4">
|
||||||
|
<div id="filter-result-js" class="text-primary-950 dark:text-white text-right"></div>
|
||||||
|
|
||||||
{% for fee in fees | sort(attribute="name") %}
|
{% for fee in fees | sort(attribute="name") %}
|
||||||
<b>{{ fee.name }}: {{ fee.sum_in_cents / 100 }}€</b><br />
|
<div {% if fee.paid %} style="background-color: green;" {% endif %} data-filterable="true" data-filter="{{ fee.name }}" class="bg-white dark:bg-primary-900 p-3 rounded-md w-full">
|
||||||
|
<div class="grid sm:grid-cols-1 gap-3">
|
||||||
|
<div style="width: 100%" class="col-span-2">
|
||||||
|
<b>{{ fee.name }}</b>
|
||||||
|
</div>
|
||||||
|
<div style="width: 100%">
|
||||||
|
{{ fee.sum_in_cents / 100 }}€:
|
||||||
|
</div>
|
||||||
|
<div style="width: 100%">
|
||||||
{% for p in fee.parts %}
|
{% for p in fee.parts %}
|
||||||
{{ p.0 }} ({{ p.1 / 100 }}€) {% if not loop.last %} + {% endif %}
|
{{ p.0 }} ({{ p.1 / 100 }}€) {% if not loop.last %} + {% endif %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
<hr />
|
</div>
|
||||||
|
{% if "admin" in loggedin_user.roles %}
|
||||||
|
<a href="/admin/user/fees/paid?{{ fee.user_ids }}">Zahlungsstatus ändern</a>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
|
|
||||||
{% endblock content %}
|
{% endblock content %}
|
||||||
|
|
||||||
|
@ -304,9 +304,9 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{# --- START Add Buttons --- #}
|
{# --- START Add Buttons --- #}
|
||||||
{% if "admin" in loggedin_user.roles or "cox" in loggedin_user.roles %}
|
{% if "planned_event" in loggedin_user.roles or "cox" in loggedin_user.roles %}
|
||||||
<div class="grid {% if "admin" in loggedin_user.roles %} grid-cols-2 {% endif %} text-center">
|
<div class="grid {% if "planned_event" in loggedin_user.roles %} grid-cols-2 {% endif %} text-center">
|
||||||
{% if "admin" in loggedin_user.roles %}
|
{% if "planned_event" in loggedin_user.roles %}
|
||||||
<a href="#" data-sidebar="true" data-trigger="sidebar" data-header="<strong>Event</strong> am {{ day.day| date(format='%d.%m.%Y') }} erstellen" data-day="{{ day.day }}" data-body="#addEventForm" class="relative inline-block w-full bg-primary-900 hover:bg-primary-950 focus:bg-primary-950 dark:bg-primary-950 text-white py-2 rounded-bl-md text-sm font-semibold">
|
<a href="#" data-sidebar="true" data-trigger="sidebar" data-header="<strong>Event</strong> am {{ day.day| date(format='%d.%m.%Y') }} erstellen" data-day="{{ day.day }}" data-body="#addEventForm" class="relative inline-block w-full bg-primary-900 hover:bg-primary-950 focus:bg-primary-950 dark:bg-primary-950 text-white py-2 rounded-bl-md text-sm font-semibold">
|
||||||
<span class="absolute inset-y-0 left-0 flex items-center pl-3">
|
<span class="absolute inset-y-0 left-0 flex items-center pl-3">
|
||||||
{% include "includes/plus-icon" %}
|
{% include "includes/plus-icon" %}
|
||||||
@ -316,7 +316,7 @@
|
|||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% if "cox" in loggedin_user.roles %}
|
{% if "cox" in loggedin_user.roles %}
|
||||||
<a href="#" data-sidebar="true" data-trigger="sidebar" data-header="<strong>Ausfahrt</strong> am {{ day.day| date(format='%d.%m.%Y') }} erstellen" data-day="{{ day.day }}" data-body="#sidebarForm" class="relative inline-block w-full py-2 text-primary-900 hover:text-primary-950 dark:bg-primary-600 dark:text-white dark:hover:bg-primary-500 dark:hover:text-white focus:text-primary-950 text-sm font-semibold bg-gray-100 hover:bg-gray-200 focus:bg-gray-200 {% if "admin" in loggedin_user.roles %} rounded-br-md {% else %} rounded-b-md {% endif %}">
|
<a href="#" data-sidebar="true" data-trigger="sidebar" data-header="<strong>Ausfahrt</strong> am {{ day.day| date(format='%d.%m.%Y') }} erstellen" data-day="{{ day.day }}" data-body="#sidebarForm" class="relative inline-block w-full py-2 text-primary-900 hover:text-primary-950 dark:bg-primary-600 dark:text-white dark:hover:bg-primary-500 dark:hover:text-white focus:text-primary-950 text-sm font-semibold bg-gray-100 hover:bg-gray-200 focus:bg-gray-200 {% if "planned_event" in loggedin_user.roles %} rounded-br-md {% else %} rounded-b-md {% endif %}">
|
||||||
<span class="absolute inset-y-0 left-0 flex items-center pl-3">
|
<span class="absolute inset-y-0 left-0 flex items-center pl-3">
|
||||||
{% include "includes/plus-icon" %}
|
{% include "includes/plus-icon" %}
|
||||||
</span>
|
</span>
|
||||||
@ -335,6 +335,6 @@
|
|||||||
{% include "forms/trip" %}
|
{% include "forms/trip" %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% if "admin" in loggedin_user.roles %}
|
{% if "planned_event" in loggedin_user.roles %}
|
||||||
{% include "forms/event" %}
|
{% include "forms/event" %}
|
||||||
{% endif %}{% endblock content %}
|
{% endif %}{% endblock content %}
|
||||||
|
Loading…
Reference in New Issue
Block a user