move todos to gitlab; write tests for /cox/remove/<id>; Closes #9
This commit is contained in:
parent
c4100e6c68
commit
bbb78cbc44
109
README.md
109
README.md
@ -1,80 +1,3 @@
|
||||
# Backend
|
||||
- [] **Create missing backend tests (see below)**
|
||||
- [] trip_details -> is_locked (default: false)
|
||||
- [] ics for registered trips
|
||||
|
||||
## New large features
|
||||
### Logbuch
|
||||
- Only layout + tests missing :-)
|
||||
|
||||
### Guest-Scheckbuch
|
||||
- guest_trip
|
||||
- guest_user_id
|
||||
- amount_trips
|
||||
- paid_to_user_id
|
||||
- guest_trip_logbook
|
||||
- guest_trip_id
|
||||
- logbook_id
|
||||
|
||||
### Bootsreservierungen
|
||||
- Confirmation required?
|
||||
- How long in advance is it possible?
|
||||
- Default reservations for some regular events (A+F, USI, ...)?
|
||||
|
||||
### Notifications
|
||||
- notifcations
|
||||
- id
|
||||
- message
|
||||
- category
|
||||
- created_at
|
||||
- confirmed_at: Option<Datetime>
|
||||
- user_id
|
||||
- link
|
||||
- ideas
|
||||
- created an event at the same datetime as you
|
||||
|
||||
### Schnupper-Pipeline
|
||||
- Mail-Adressen von Interessierten dauerhaft entgegennehmen
|
||||
- Termin ausgemacht -> Interessierte kontaktieren
|
||||
- X Personen können teilnehmen (bis zu 3(?) pro Person erlauben (Familie)?)
|
||||
- Automatisch Bestätigung bei Anmeldung schicken, mit Detail-Infos
|
||||
- Ein paar Tage vorher Erinnerungs-Mail ausschicken
|
||||
- Anmeldungen können manuell wieder gelöscht werden
|
||||
- Es gibt Liste mit aktuellen Anmeldungen
|
||||
|
||||
### Ergochallenge
|
||||
- Bilder + Dateneingabe
|
||||
- Automatische Mail senden
|
||||
|
||||
## Backlog (i.e. don't work on this now)
|
||||
### Sync w/ nextcloud
|
||||
- remove most fields (names, ...) from users and add uid
|
||||
- create user_nextcloud table; to be re-created every day(?)
|
||||
|
||||
user
|
||||
- UID
|
||||
- pw
|
||||
- last_access
|
||||
|
||||
user_details
|
||||
- UID
|
||||
- fn (formatted name)
|
||||
- is_cox (if CATEGORIES = {Steuerleute, Bootsführer})
|
||||
- is_admin (if CATEGORIES = Admin)
|
||||
- is_guest (if person not in nextcloud)
|
||||
|
||||
### Misc
|
||||
- [] Don't show events if time > 1h(?) ago
|
||||
- [] exactly same time -> deny registration
|
||||
- [] automatically add regular planned trip
|
||||
- [] same day+time: aggregate stats (x people, of which y cox and z rower)
|
||||
- [] Lock trip; noone can register anymore
|
||||
- [] on delete cascade doesn't work; e.g. created planned_event/trip + delete it -> trip_details entry still there!
|
||||
- [] allow users to add u2f key
|
||||
- [] Möglichkeiten für Bootseinteilungen bei planned_events anzeigen
|
||||
|
||||
|
||||
|
||||
# Frontend Process
|
||||
´cd frontend´
|
||||
´npm install´
|
||||
@ -82,7 +5,6 @@ user_details
|
||||
|
||||
# Notes / Bugfixes
|
||||
## Frontend
|
||||
- [] add UI for `trip_type`
|
||||
- [] support esc to close sidebar
|
||||
- [] after an hour(?) of inactivity -> show large popup w/ "maybe old data (ignore) (reload page)" (ignore bc maybe use is actively doing something -> don't throw input away!)
|
||||
|
||||
@ -90,34 +12,3 @@ user_details
|
||||
# Nice to have
|
||||
## Frontend
|
||||
- [] my trips for cox
|
||||
|
||||
# Missing backend tests
|
||||
|
||||
- [x] (index) GET /
|
||||
- [x] (faq) GET /faq
|
||||
- [x] (cal) GET /cal
|
||||
- [x] (FileServer: svelte/build) GET /<path..>
|
||||
- [x] (join) GET /join/<trip_details_id>
|
||||
- [x] (remove) GET /remove/<trip_details_id>
|
||||
- [x] (create) POST /cox/trip
|
||||
- [x] (update) POST /cox/trip/<trip_id>
|
||||
- [x] (join) GET /cox/join/<planned_event_id>
|
||||
- [ ] (remove) GET /cox/remove/<planned_event_id>
|
||||
- [ ] (remove_trip) GET /cox/remove/trip/<trip_id>
|
||||
- [ ] (index) GET /auth/
|
||||
- [ ] (login) POST /auth/
|
||||
- [ ] (logout) GET /auth/logout
|
||||
- [ ] (updatepw) POST /auth/set-pw
|
||||
- [ ] (setpw) GET /auth/set-pw/<userid>
|
||||
- [ ] (rss) GET /admin/rss?<key>
|
||||
- [ ] (index) GET /admin/user
|
||||
- [ ] (update) POST /admin/user
|
||||
- [ ] (create) POST /admin/planned-event
|
||||
- [ ] (update) PUT /admin/planned-event
|
||||
- [ ] (create) POST /admin/user/new
|
||||
- [ ] (delete) GET /admin/user/<user>/delete
|
||||
- [ ] (resetpw) GET /admin/user/<user>/reset-pw
|
||||
- [ ] (delete) GET /admin/planned-event/<id>/delete
|
||||
- [ ] (FileServer: static/) GET /public/<path..> [10]
|
||||
- [ ] (login) POST /api/login/
|
||||
- [ ] /tera/admin/boat.rs
|
||||
|
@ -176,7 +176,7 @@ FROM user_trip WHERE trip_details_id = (SELECT trip_details_id FROM trip WHERE i
|
||||
db: &SqlitePool,
|
||||
cox: &CoxUser,
|
||||
planned_event: &PlannedEvent,
|
||||
) {
|
||||
) -> bool {
|
||||
sqlx::query!(
|
||||
"DELETE FROM trip WHERE cox_id = ? AND planned_event_id = ?",
|
||||
cox.id,
|
||||
@ -184,7 +184,9 @@ FROM user_trip WHERE trip_details_id = (SELECT trip_details_id FROM trip WHERE i
|
||||
)
|
||||
.execute(db)
|
||||
.await
|
||||
.unwrap(); //TODO: handle case where cox is not registered
|
||||
.unwrap()
|
||||
.rows_affected()
|
||||
> 0
|
||||
}
|
||||
|
||||
pub(crate) async fn delete(
|
||||
|
103
src/tera/cox.rs
103
src/tera/cox.rs
@ -129,8 +129,7 @@ async fn remove_trip(db: &State<SqlitePool>, trip_id: i64, cox: CoxUser) -> Flas
|
||||
#[get("/remove/<planned_event_id>")]
|
||||
async fn remove(db: &State<SqlitePool>, planned_event_id: i64, cox: CoxUser) -> Flash<Redirect> {
|
||||
if let Some(planned_event) = PlannedEvent::find_by_id(db, planned_event_id).await {
|
||||
Trip::delete_by_planned_event(db, &cox, &planned_event).await;
|
||||
|
||||
if Trip::delete_by_planned_event(db, &cox, &planned_event).await {
|
||||
Log::create(
|
||||
db,
|
||||
format!(
|
||||
@ -141,6 +140,9 @@ async fn remove(db: &State<SqlitePool>, planned_event_id: i64, cox: CoxUser) ->
|
||||
.await;
|
||||
|
||||
Flash::success(Redirect::to("/"), "Erfolgreich abgemeldet!")
|
||||
} else {
|
||||
Flash::error(Redirect::to("/"), "Steuermann hilft nicht aus...")
|
||||
}
|
||||
} else {
|
||||
Flash::error(Redirect::to("/"), "Planned_event does not exist.")
|
||||
}
|
||||
@ -425,4 +427,101 @@ mod test {
|
||||
|
||||
assert_eq!(flash_cookie.value(), "5:errorEvent gibt's nicht");
|
||||
}
|
||||
|
||||
#[sqlx::test]
|
||||
fn test_remove() {
|
||||
let db = testdb!();
|
||||
|
||||
let rocket = rocket::build().manage(db.clone());
|
||||
let rocket = crate::tera::config(rocket);
|
||||
|
||||
let client = Client::tracked(rocket).await.unwrap();
|
||||
let login = client
|
||||
.post("/auth")
|
||||
.header(ContentType::Form) // Set the content type to form
|
||||
.body("name=cox&password=cox"); // Add the form data to the request body;
|
||||
login.dispatch().await;
|
||||
|
||||
let req = client.get("/cox/join/1");
|
||||
let response = req.dispatch().await;
|
||||
|
||||
let flash_cookie = response
|
||||
.cookies()
|
||||
.get("_flash")
|
||||
.expect("Expected flash cookie");
|
||||
|
||||
assert_eq!(flash_cookie.value(), "7:successDanke für's helfen!");
|
||||
|
||||
let req = client.get("/cox/join/1");
|
||||
let response = req.dispatch().await;
|
||||
|
||||
let req = client.get("/cox/remove/1");
|
||||
let response = req.dispatch().await;
|
||||
|
||||
assert_eq!(response.status(), Status::SeeOther);
|
||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
||||
|
||||
let flash_cookie = response
|
||||
.cookies()
|
||||
.get("_flash")
|
||||
.expect("Expected flash cookie");
|
||||
|
||||
assert_eq!(flash_cookie.value(), "7:successErfolgreich abgemeldet!");
|
||||
}
|
||||
|
||||
#[sqlx::test]
|
||||
fn test_remove_wrong_id() {
|
||||
let db = testdb!();
|
||||
|
||||
let rocket = rocket::build().manage(db.clone());
|
||||
let rocket = crate::tera::config(rocket);
|
||||
|
||||
let client = Client::tracked(rocket).await.unwrap();
|
||||
let login = client
|
||||
.post("/auth")
|
||||
.header(ContentType::Form) // Set the content type to form
|
||||
.body("name=cox&password=cox"); // Add the form data to the request body;
|
||||
login.dispatch().await;
|
||||
|
||||
let req = client.get("/cox/remove/999");
|
||||
let response = req.dispatch().await;
|
||||
|
||||
assert_eq!(response.status(), Status::SeeOther);
|
||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
||||
|
||||
let flash_cookie = response
|
||||
.cookies()
|
||||
.get("_flash")
|
||||
.expect("Expected flash cookie");
|
||||
|
||||
assert_eq!(flash_cookie.value(), "5:errorPlanned_event does not exist.");
|
||||
}
|
||||
|
||||
#[sqlx::test]
|
||||
fn test_remove_cox_not_participating() {
|
||||
let db = testdb!();
|
||||
|
||||
let rocket = rocket::build().manage(db.clone());
|
||||
let rocket = crate::tera::config(rocket);
|
||||
|
||||
let client = Client::tracked(rocket).await.unwrap();
|
||||
let login = client
|
||||
.post("/auth")
|
||||
.header(ContentType::Form) // Set the content type to form
|
||||
.body("name=cox&password=cox"); // Add the form data to the request body;
|
||||
login.dispatch().await;
|
||||
|
||||
let req = client.get("/cox/remove/1");
|
||||
let response = req.dispatch().await;
|
||||
|
||||
assert_eq!(response.status(), Status::SeeOther);
|
||||
assert_eq!(response.headers().get("Location").next(), Some("/"));
|
||||
|
||||
let flash_cookie = response
|
||||
.cookies()
|
||||
.get("_flash")
|
||||
.expect("Expected flash cookie");
|
||||
|
||||
assert_eq!(flash_cookie.value(), "5:errorSteuermann hilft nicht aus...");
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user