functionally completed user editing

This commit is contained in:
z1glr
2025-01-12 02:01:14 +00:00
parent ac6bf24d57
commit 0685283007
18 changed files with 410 additions and 132 deletions

View File

@@ -2,7 +2,6 @@ package availabilities
import ( import (
"github.com/johannesbuehl/golunteer/backend/pkg/db" "github.com/johannesbuehl/golunteer/backend/pkg/db"
"github.com/johannesbuehl/golunteer/backend/pkg/db/users"
) )
type eventAvailabilities struct { type eventAvailabilities struct {
@@ -23,11 +22,9 @@ func Event(eventID int) (map[string]string, error) {
// get the availabilities // get the availabilities
if availabilitiesMap, err := Keys(); err != nil { if availabilitiesMap, err := Keys(); err != nil {
return nil, err return nil, err
} else if usersMap, err := users.Get(); err != nil {
return nil, err
} else { } else {
for _, a := range availabilitiesRows { for _, a := range availabilitiesRows {
eventAvailabilities[usersMap[a.UserName].Name] = availabilitiesMap[a.AvailabilityID].Text eventAvailabilities[a.UserName] = availabilitiesMap[a.AvailabilityID].Text
} }
return eventAvailabilities, nil return eventAvailabilities, nil

View File

@@ -1,42 +1,45 @@
package users package users
import ( import (
"fmt"
"time"
"github.com/google/uuid" "github.com/google/uuid"
cache "github.com/jfarleyx/go-simple-cache"
"github.com/johannesbuehl/golunteer/backend/pkg/db" "github.com/johannesbuehl/golunteer/backend/pkg/db"
"golang.org/x/crypto/bcrypt" "golang.org/x/crypto/bcrypt"
) )
type User struct { type User struct {
Name string `db:"name"` Name string `db:"name" json:"userName"`
Password []byte `db:"password"` Admin bool `db:"admin" json:"admin"`
TokenID string `db:"tokenID"`
Admin bool `db:"admin"`
} }
var c *cache.Cache
// hashes a password // hashes a password
func hashPassword(password string) ([]byte, error) { func hashPassword(password string) ([]byte, error) {
return bcrypt.GenerateFromPassword([]byte(password), bcrypt.DefaultCost) return bcrypt.GenerateFromPassword([]byte(password), bcrypt.DefaultCost)
} }
func Get() (map[string]User, error) { func Get() ([]User, error) {
if users, hit := c.Get("users"); !hit { // get the users from the database
refresh() var users []User
return nil, fmt.Errorf("users not cached") if err := db.DB.Select(&users, "SELECT name, admin FROM USERS"); err != nil {
return nil, err
} else { } else {
return users.(map[string]User), nil return users, nil
} }
} }
func TokenID(userName string) (string, error) {
var dbResult struct {
TokenID string `db:"tokenID"`
}
err := db.DB.Get(&dbResult, "SELECT tokenID FROM USERS WHERE name = ?", userName)
return dbResult.TokenID, err
}
type UserAdd struct { type UserAdd struct {
UserName string `json:"userName" validate:"required" db:"userName"` UserName string `json:"userName" validate:"required" db:"userName"`
Password string `json:"password" validate:"required,min=12"` Password string `json:"password" validate:"required,min=12,max=64"`
Admin bool `json:"admin" db:"admin"` Admin bool `json:"admin" db:"admin"`
} }
@@ -55,13 +58,9 @@ func Add(user UserAdd) error {
TokenID: uuid.NewString(), TokenID: uuid.NewString(),
} }
if _, err := db.DB.NamedExec("INSERT INTO USERS (name, password, admin, tokenID) VALUES (:userName, :password, :admin, :tokenID)", insertUser); err != nil { _, err := db.DB.NamedExec("INSERT INTO USERS (name, password, admin, tokenID) VALUES (:userName, :password, :admin, :tokenID)", insertUser)
return err
} else {
refresh()
return nil return err
}
} }
} }
@@ -88,33 +87,19 @@ func ChangePassword(user UserChangePassword) (string, error) {
if _, err := db.DB.NamedExec("UPDATE USERS SET tokenID = :tokenID, password = :password WHERE name = :userName", execStruct); err != nil { if _, err := db.DB.NamedExec("UPDATE USERS SET tokenID = :tokenID, password = :password WHERE name = :userName", execStruct); err != nil {
return "", err return "", err
} else { } else {
refresh()
return execStruct.TokenID, nil return execStruct.TokenID, nil
} }
} }
} }
func refresh() { func ChangeName(userName, newName string) error {
// get the usersRaw from the database _, err := db.DB.Exec("UPDATE USERS SET name = ? WHERE name = ?", newName, userName)
var usersRaw []User
if err := db.DB.Select(&usersRaw, "SELECT * FROM USERS"); err == nil { return err
// convert the result in a map
users := map[string]User{}
for _, user := range usersRaw {
users[user.Name] = user
}
c.Set("users", users)
}
} }
func init() { func SetAdmin(userName string, admin bool) error {
c = cache.New(24 * time.Hour) _, err := db.DB.Exec("UPDATE USERS SET admin = ? WHERE name = ?", admin, userName)
c.OnExpired(refresh) return err
refresh()
} }

View File

@@ -15,21 +15,23 @@ func handleWelcome(c *fiber.Ctx) error {
Admin: false, Admin: false,
} }
if user, err := checkUser(c); err != nil { args := HandlerArgs{C: c}
if loggedIn, err := args.checkUser(); err != nil {
response.Status = fiber.StatusInternalServerError response.Status = fiber.StatusInternalServerError
logger.Warn().Msgf("can't check user: %v", err) logger.Warn().Msgf("can't check user: %v", err)
} else if user == nil { } else if !loggedIn {
response.Status = fiber.StatusNoContent response.Status = fiber.StatusUnauthorized
logger.Debug().Msgf("user not authorized") logger.Debug().Msgf("user not authorized")
} else { } else {
response.Data = UserChecked{ response.Data = UserChecked{
UserName: user.UserName, UserName: args.User.UserName,
Admin: user.Admin, Admin: args.User.Admin,
} }
logger.Debug().Msgf("welcomed user %q", user.UserName) logger.Debug().Msgf("welcomed user %q", args.User.UserName)
} }
return response.send(c) return response.send(c)
@@ -40,6 +42,8 @@ const messageWrongLogin = "Unkown user or wrong password"
func handleLogin(c *fiber.Ctx) error { func handleLogin(c *fiber.Ctx) error {
logger.Debug().Msgf("HTTP %s request: %q", c.Method(), c.OriginalURL()) logger.Debug().Msgf("HTTP %s request: %q", c.Method(), c.OriginalURL())
args := HandlerArgs{C: c}
// extract username and password from the request // extract username and password from the request
requestBody := struct { requestBody := struct {
Username string `json:"userName" validate:"required"` Username string `json:"userName" validate:"required"`
@@ -48,7 +52,7 @@ func handleLogin(c *fiber.Ctx) error {
var response responseMessage var response responseMessage
if err := c.BodyParser(&requestBody); err != nil { if err := args.C.BodyParser(&requestBody); err != nil {
logger.Debug().Msgf("can't parse login-body: %v", err) logger.Debug().Msgf("can't parse login-body: %v", err)
response.Status = fiber.StatusBadRequest response.Status = fiber.StatusBadRequest
@@ -79,7 +83,7 @@ func handleLogin(c *fiber.Ctx) error {
response.Status = fiber.StatusInternalServerError response.Status = fiber.StatusInternalServerError
logger.Error().Msgf("can't create JWT: %v", err) logger.Error().Msgf("can't create JWT: %v", err)
} else { } else {
setSessionCookie(c, &jwt) args.setSessionCookie(&jwt)
response.Data = UserChecked{ response.Data = UserChecked{
UserName: requestBody.Username, UserName: requestBody.Username,
@@ -92,14 +96,18 @@ func handleLogin(c *fiber.Ctx) error {
} }
} }
return response.send(c) return response.send(args.C)
} }
// handles logout-requests // handles logout-requests
func handleLogout(c *fiber.Ctx) error { func handleLogout(c *fiber.Ctx) error {
logger.Debug().Msgf("HTTP %s request: %q", c.Method(), c.OriginalURL()) logger.Debug().Msgf("HTTP %s request: %q", c.Method(), c.OriginalURL())
removeSessionCookie(c) args := HandlerArgs{
C: c,
}
args.removeSessionCookie()
return responseMessage{}.send(c) return responseMessage{}.send(c)
} }

View File

@@ -79,6 +79,7 @@ func init() {
"events/availabilities": getEventsAvailabilities, "events/availabilities": getEventsAvailabilities,
"events/user/pending": getEventsUserPending, "events/user/pending": getEventsUserPending,
"tasks": getTasks, "tasks": getTasks,
"users": getUsers,
}, },
"POST": { "POST": {
"events": postEvent, "events": postEvent,
@@ -86,6 +87,7 @@ func init() {
}, },
"PATCH": { "PATCH": {
"users/password": patchPassword, "users/password": patchPassword,
"users": patchUser,
}, },
"DELETE": { "DELETE": {
"event": deleteEvent, "event": deleteEvent,
@@ -104,24 +106,24 @@ func init() {
logger.Debug().Msgf("HTTP %s request: %q", c.Method(), c.OriginalURL()) logger.Debug().Msgf("HTTP %s request: %q", c.Method(), c.OriginalURL())
var response responseMessage var response responseMessage
args := HandlerArgs{
C: c,
}
if user, err := checkUser(c); err != nil { if loggedIn, err := args.checkUser(); err != nil {
response = responseMessage{ response = responseMessage{
Status: fiber.StatusBadRequest, Status: fiber.StatusBadRequest,
} }
logger.Error().Msgf("can't check user: %v", err) logger.Error().Msgf("can't check user: %v", err)
} else if user == nil { } else if !loggedIn {
response = responseMessage{ response = responseMessage{
Status: fiber.StatusNoContent, Status: fiber.StatusUnauthorized,
} }
logger.Log().Msgf("user not authorized") logger.Log().Msgf("user not authorized")
} else { } else {
response = handler(HandlerArgs{ response = handler(args)
C: c,
User: *user,
})
} }
return response.send(c) return response.send(c)
@@ -137,16 +139,16 @@ func Listen() {
fmt.Println(err) fmt.Println(err)
} }
func setSessionCookie(c *fiber.Ctx, jwt *string) { func (args HandlerArgs) setSessionCookie(jwt *string) {
var value string var value string
if jwt == nil { if jwt == nil {
value = c.Cookies("session") value = args.C.Cookies("session")
} else { } else {
value = *jwt value = *jwt
} }
c.Cookie(&fiber.Cookie{ args.C.Cookie(&fiber.Cookie{
Name: "session", Name: "session",
Value: value, Value: value,
HTTPOnly: true, HTTPOnly: true,
@@ -156,8 +158,8 @@ func setSessionCookie(c *fiber.Ctx, jwt *string) {
} }
// removes the session-coockie from a request // removes the session-coockie from a request
func removeSessionCookie(c *fiber.Ctx) { func (args HandlerArgs) removeSessionCookie() {
c.Cookie(&fiber.Cookie{ args.C.Cookie(&fiber.Cookie{
Name: "session", Name: "session",
Value: "", Value: "",
HTTPOnly: true, HTTPOnly: true,
@@ -219,11 +221,11 @@ type UserChecked struct {
} }
// checks wether the request is from a valid user // checks wether the request is from a valid user
func checkUser(c *fiber.Ctx) (*UserChecked, error) { func (args *HandlerArgs) checkUser() (bool, error) {
userName, tokenID, err := extractJWT(c) userName, tokenID, err := extractJWT(args.C)
if err != nil { if err != nil {
return nil, nil return false, nil
} }
var dbResult struct { var dbResult struct {
@@ -232,19 +234,21 @@ func checkUser(c *fiber.Ctx) (*UserChecked, error) {
} }
// retrieve the user from the database // retrieve the user from the database
if err := db.DB.QueryRowx("SELECT tokenID, admin FROM USERS WHERE name = ?", userName).StructScan(&dbResult); err != nil { if err := db.DB.Get(&dbResult, "SELECT tokenID, admin FROM USERS WHERE name = ?", userName); err != nil {
return nil, err return false, err
// if the tokenID is valid, the user is authorized // if the tokenID is valid, the user is authorized
} else if dbResult.TokenID != tokenID { } else if dbResult.TokenID != tokenID {
return nil, err return false, nil
} else { } else {
// reset the expiration of the cookie // reset the expiration of the cookie
setSessionCookie(c, nil) args.setSessionCookie(nil)
return &UserChecked{ args.User = UserChecked{
UserName: userName, UserName: userName,
Admin: dbResult.Admin, Admin: dbResult.Admin,
}, err }
return true, nil
} }
} }

View File

@@ -5,6 +5,25 @@ import (
"github.com/johannesbuehl/golunteer/backend/pkg/db/users" "github.com/johannesbuehl/golunteer/backend/pkg/db/users"
) )
func getUsers(args HandlerArgs) responseMessage {
response := responseMessage{}
// check admin
if !args.User.Admin {
response.Status = fiber.StatusForbidden
logger.Log().Msgf("user is no admin")
} else if users, err := users.Get(); err != nil {
response.Status = fiber.StatusInternalServerError
logger.Error().Msgf("can't get users: %v", err)
} else {
response.Data = users
}
return response
}
func postUser(args HandlerArgs) responseMessage { func postUser(args HandlerArgs) responseMessage {
response := responseMessage{} response := responseMessage{}
@@ -63,12 +82,108 @@ func patchPassword(args HandlerArgs) responseMessage {
// if something failed, remove the current session-cookie // if something failed, remove the current session-cookie
}); err != nil { }); err != nil {
removeSessionCookie(args.C) args.removeSessionCookie()
// set the new session-cookie // set the new session-cookie
} else { } else {
// update the token in the session-cookie // update the token in the session-cookie
setSessionCookie(args.C, &jwt) args.setSessionCookie(&jwt)
}
}
return response
}
func patchUser(args HandlerArgs) responseMessage {
response := responseMessage{}
// check admin
if !args.User.Admin {
response.Status = fiber.StatusForbidden
logger.Log().Msgf("user is no admin")
} else {
// parse the body
var body struct {
users.UserAdd
NewName string `json:"newName"`
}
if err := args.C.BodyParser(&body); err != nil {
response.Status = fiber.StatusBadRequest
logger.Log().Msgf("can't parse body: %v", err)
// prevent to demoting self from admin
} else if !body.Admin && body.UserName == args.User.UserName {
response.Status = fiber.StatusBadRequest
logger.Warn().Msgf("can't demote self from admin")
} else {
// check for an empty user-name
if len(body.UserName) == 0 {
response.Status = fiber.StatusBadRequest
logger.Warn().Msgf("username is empty")
// if the password has length 0 assume the password shouldn't be changed
} else {
if len(body.Password) > 0 {
// create a password-change-struct and validate it. use the old user-name, since the new isn't stored yet
usePasswordChange := users.UserChangePassword{
UserName: body.UserName,
Password: body.Password,
}
if _, err = users.ChangePassword(usePasswordChange); err != nil {
response.Status = fiber.StatusInternalServerError
logger.Error().Msgf("can't change password: %v", err)
return response
}
}
// only change the name, if it differs
if body.NewName != body.UserName {
if err := users.ChangeName(body.UserName, body.NewName); err != nil {
response.Status = fiber.StatusInternalServerError
logger.Error().Msgf("can't change user-name: %v", err)
return response
}
}
// set the admin-status
if err := users.SetAdmin(body.NewName, body.Admin); err != nil {
response.Status = fiber.StatusInternalServerError
logger.Error().Msgf("updating admin-status failed: %v", err)
} else {
// if we modified ourself, update the session-cookie
if body.UserName == args.User.UserName {
// get the tokenID
if tokenID, err := users.TokenID(body.NewName); err != nil {
response.Status = fiber.StatusInternalServerError
logger.Error().Msgf("can't get tokenID: %v", err)
} else if jwt, err := config.SignJWT(JWTPayload{
UserName: body.NewName,
TokenID: tokenID,
}); err != nil {
response.Status = fiber.StatusInternalServerError
logger.Error().Msgf("JWT-signing failed: %v", err)
// remove the session-cookie
args.removeSessionCookie()
} else {
args.setSessionCookie(&jwt)
}
}
}
}
} }
} }

View File

@@ -10,7 +10,7 @@ const nextConfig: NextConfig = {
fallback: [ fallback: [
{ {
source: "/api/:path*", source: "/api/:path*",
destination: "http://golunteer-frontend:8080/api/:path*", destination: "http://golunteer-backend:8080/api/:path*",
}, },
], ],
}), }),

View File

@@ -2,7 +2,6 @@
import { create } from "zustand"; import { create } from "zustand";
import { persist } from "zustand/middleware"; import { persist } from "zustand/middleware";
import { apiCall } from "./lib";
export interface EventData { export interface EventData {
id: number; id: number;
@@ -17,48 +16,29 @@ export interface User {
} }
interface Zustand { interface Zustand {
events: EventData[];
pendingEvents: number;
user: User | null; user: User | null;
setEvents: (events: EventData[]) => void;
reset: (zustand?: Partial<Zustand>) => void; reset: (zustand?: Partial<Zustand>) => void;
getPendingEvents: () => Promise<void>;
} }
const initialState = { const initialState = {
events: [],
user: null, user: null,
pendingEvents: 0,
}; };
const zustand = create<Zustand>()( const zustand = create<Zustand>()(
persist( persist(
(set) => ({ (set) => ({
...initialState, ...initialState,
setEvents: (events) => set({ events }),
reset: (newZustand) => reset: (newZustand) =>
set({ set({
...initialState, ...initialState,
...newZustand, ...newZustand,
}), }),
getPendingEvents: async () => {
const result = await apiCall<{ pendingEvents: number }>(
"GET",
"events/user/pending",
);
if (result.ok) {
const resultData = await result.json();
set(() => ({ pendingEvents: resultData }));
}
},
}), }),
{ {
name: "golunteer-storage", name: "golunteer-storage",
partialize: (state) => partialize: (state) =>
Object.fromEntries( Object.fromEntries(
Object.entries(state).filter(([key]) => !["events"].includes(key)), Object.entries(state).filter(([key]) => ["user"].includes(key)),
), ),
}, },
), ),

View File

@@ -21,12 +21,25 @@ import {
} from "@nextui-org/react"; } from "@nextui-org/react";
import zustand from "@/Zustand"; import zustand from "@/Zustand";
import { SiteLink } from "./layout"; import { SiteLink } from "./layout";
import React from "react"; import React, { useEffect, useState } from "react";
export default function Header({ sites }: { sites: SiteLink[] }) { export default function Header({ sites }: { sites: SiteLink[] }) {
const router = useRouter(); const router = useRouter();
const user = zustand((state) => state.user); const user = zustand((state) => state.user);
const pendingEvents = zustand((state) => state.pendingEvents); const [pendingEvents, setPendingEvents] = useState(0);
useEffect(() => {
(async () => {
const result = await apiCall<{ pendingEvents: number }>(
"GET",
"events/user/pending",
);
if (result.ok) {
setPendingEvents(await result.json());
}
})();
}, []);
const pathname = usePathname(); const pathname = usePathname();

View File

@@ -18,6 +18,7 @@ export default function Main({ children }: { children: React.ReactNode }) {
const router = useRouter(); const router = useRouter();
const pathname = usePathname(); const pathname = usePathname();
const user = zustand((state) => state.user);
useEffect(() => { useEffect(() => {
void (async () => { void (async () => {
@@ -34,14 +35,13 @@ export default function Main({ children }: { children: React.ReactNode }) {
const response = await welcomeResult.json(); const response = await welcomeResult.json();
if (response.userName !== undefined && response.userName !== "") { if (response.userName !== undefined && response.userName !== "") {
void zustand.getState().getPendingEvents();
zustand.getState().reset({ user: response }); zustand.getState().reset({ user: response });
loggedIn = true; loggedIn = true;
} }
} catch {} } catch {}
} else { } else {
zustand.getState().reset();
} }
} else { } else {
loggedIn = true; loggedIn = true;
@@ -62,7 +62,7 @@ export default function Main({ children }: { children: React.ReactNode }) {
} }
} }
})(); })();
}, [pathname, router]); }, [pathname, router, user]);
switch (auth) { switch (auth) {
case AuthState.Loading: case AuthState.Loading:

View File

@@ -1,11 +1,8 @@
"use client"; "use client";
import { Add } from "@carbon/icons-react"; import { Add } from "@carbon/icons-react";
import Event from "../components/Event/Event";
import { useState } from "react"; import { useState } from "react";
import AddEvent from "../components/Event/AddEvent"; import AddEvent from "../components/Event/AddEvent";
import zustand from "../Zustand";
import AssignmentTable from "@/components/Event/AssignmentTable";
import { Button } from "@nextui-org/react"; import { Button } from "@nextui-org/react";
export default function EventVolunteer() { export default function EventVolunteer() {
@@ -14,13 +11,7 @@ export default function EventVolunteer() {
return ( return (
<div className="relative flex-1"> <div className="relative flex-1">
<h2 className="mb-4 text-center text-4xl">Overview</h2> <h2 className="mb-4 text-center text-4xl">Overview</h2>
<div className="flex flex-wrap justify-center gap-4"> <div className="flex flex-wrap justify-center gap-4"></div>
{zustand.getState().events.map((ee) => (
<Event key={ee.id} event={ee}>
<AssignmentTable tasks={ee.tasks} />
</Event>
))}
</div>
<Button <Button
color="primary" color="primary"

View File

@@ -25,7 +25,6 @@ export default function AddUser(props: {
onOpenChange={props.onOpenChange} onOpenChange={props.onOpenChange}
shadow={"none" as "sm"} shadow={"none" as "sm"}
backdrop="blur" backdrop="blur"
className="bg-accent-5"
> >
<ModalContent> <ModalContent>
<ModalHeader> <ModalHeader>

View File

@@ -0,0 +1,137 @@
import {
apiCall,
classNames,
vaidatePassword as validatePassword,
} from "@/lib";
import zustand, { User } from "@/Zustand";
import {
Button,
Checkbox,
Form,
Input,
Modal,
ModalBody,
ModalContent,
ModalFooter,
ModalHeader,
} from "@nextui-org/react";
import { FormEvent, useEffect, useState } from "react";
export default function EditUser(props: {
isOpen: boolean;
user?: User;
onOpenChange: (isOpen: boolean) => void;
onSuccess: () => void;
}) {
const [name, setName] = useState(props.user?.userName);
const [admin, setAdmin] = useState(props.user?.admin);
const [password, setPassword] = useState("");
const pwErrors = validatePassword(password);
// set the states on value changes
useEffect(() => {
if (props.user !== undefined) {
setName(props.user.userName);
setAdmin(props.user.admin);
// reset the password
setPassword("");
}
}, [props.user]);
// update the user in the backend
async function updateUser(e: FormEvent<HTMLFormElement>) {
const formData = Object.fromEntries(new FormData(e.currentTarget));
const data = {
...formData,
userName: props.user?.userName,
admin: formData.admin !== undefined,
};
// if we modify ourself, set admin to true since it isn't included in the form data because the checkbox is disabled
data.admin ||= props.user?.userName === zustand.getState().user?.userName;
const result = await apiCall("PATCH", "users", undefined, data);
if (result.ok) {
// if we updated ourself
if (props.user?.userName === zustand.getState().user?.userName) {
zustand.setState({ user: null });
}
props.onSuccess();
}
}
return (
<Modal isOpen={props.isOpen} onOpenChange={props.onOpenChange}>
{props.user !== undefined ? (
<ModalContent>
<ModalHeader>
<h1 className="text-2xl">
Edit User{" "}
<span className="font-numbers font-normal italic">
{props.user.userName}
</span>
</h1>
</ModalHeader>
<Form
validationBehavior="native"
onSubmit={(e) => {
e.preventDefault();
updateUser(e);
}}
>
<ModalBody className="w-full">
<Input
label="Name"
color={name !== props.user.userName ? "warning" : "default"}
name="newName"
value={name}
onValueChange={setName}
/>
<Input
label="Password"
color={password.length > 0 ? "warning" : "default"}
name="password"
value={password}
onValueChange={setPassword}
isInvalid={password.length > 0 && pwErrors.length > 0}
errorMessage={
<ul>
{pwErrors.map((e, ii) => (
<li key={ii}>{e}</li>
))}
</ul>
}
/>
<Checkbox
name="admin"
color={admin !== props.user.admin ? "warning" : "primary"}
isDisabled={
props.user.userName === zustand.getState().user?.userName
}
isSelected={admin}
onValueChange={setAdmin}
classNames={{
label: classNames({
"text-warning": admin !== props.user.admin,
}),
}}
>
Admin
</Checkbox>
</ModalBody>
<ModalFooter>
<Button type="submit" color="primary">
Update
</Button>
</ModalFooter>
</Form>
</ModalContent>
) : null}
</Modal>
);
}

View File

@@ -9,22 +9,33 @@ import {
TableColumn, TableColumn,
TableHeader, TableHeader,
TableRow, TableRow,
Tooltip,
} from "@nextui-org/react"; } from "@nextui-org/react";
import { useAsyncList } from "@react-stately/data"; import { useAsyncList } from "@react-stately/data";
import { FormEvent, useState } from "react"; import { FormEvent, useState } from "react";
import AddUser from "./AddUser"; import AddUser from "./AddUser";
import { Edit } from "@carbon/icons-react";
import EditUser from "./EditUser";
export default function Users() { export default function Users() {
const [showAddUser, setShowAddUser] = useState(false); const [showAddUser, setShowAddUser] = useState(false);
const [editUser, setEditUser] = useState<User | undefined>();
const users = useAsyncList<User>({ const users = useAsyncList<User>({
async load() { async load() {
return { const result = await apiCall("GET", "users");
items: [
{ userName: "admin", admin: true }, if (result.ok) {
{ userName: "foo", admin: false }, const users = (await result.json()) as User[];
{ userName: "bar", admin: true },
], return {
}; items: users,
};
} else {
return {
items: [],
};
}
}, },
async sort({ items, sortDescriptor }) { async sort({ items, sortDescriptor }) {
return { return {
@@ -67,6 +78,7 @@ export default function Users() {
} }
} }
// content above the user-tabel
const topContent = ( const topContent = (
<> <>
<Button onPress={() => setShowAddUser(true)}>Add User</Button> <Button onPress={() => setShowAddUser(true)}>Add User</Button>
@@ -90,6 +102,7 @@ export default function Users() {
<TableColumn allowsSorting key="admin"> <TableColumn allowsSorting key="admin">
Admin Admin
</TableColumn> </TableColumn>
<TableColumn key="edit">Edit</TableColumn>
</TableHeader> </TableHeader>
<TableBody items={users.items}> <TableBody items={users.items}>
{(user) => ( {(user) => (
@@ -98,6 +111,18 @@ export default function Users() {
<TableCell> <TableCell>
<Checkbox isSelected={user.admin} /> <Checkbox isSelected={user.admin} />
</TableCell> </TableCell>
<TableCell>
<Button
isIconOnly
variant="light"
size="sm"
onPress={() => setEditUser(user)}
>
<Tooltip content="Edit event">
<Edit />
</Tooltip>
</Button>
</TableCell>
</TableRow> </TableRow>
)} )}
</TableBody> </TableBody>
@@ -108,6 +133,17 @@ export default function Users() {
onOpenChange={setShowAddUser} onOpenChange={setShowAddUser}
onSubmit={(e) => void addUser(e)} onSubmit={(e) => void addUser(e)}
/> />
<EditUser
isOpen={editUser !== undefined}
user={editUser}
onOpenChange={(isOpen) =>
!isOpen ? setEditUser(undefined) : undefined
}
onSuccess={() => {
users.reload();
setEditUser(undefined);
}}
/>
</div> </div>
); );
} }

View File

@@ -15,8 +15,6 @@ export default function Events() {
useEffect(() => { useEffect(() => {
(async () => { (async () => {
console.debug("query");
const data = await apiCall<EventData[]>("GET", "events/assignments"); const data = await apiCall<EventData[]>("GET", "events/assignments");
if (data.ok) { if (data.ok) {

View File

@@ -27,9 +27,6 @@ export default function Login() {
// add the user-info to the zustand // add the user-info to the zustand
zustand.getState().reset({ user: await result.json() }); zustand.getState().reset({ user: await result.json() });
// retrieve the notifications
await zustand.getState().getPendingEvents();
// redirect to the home-page // redirect to the home-page
router.push("/"); router.push("/");
} else { } else {

View File

@@ -33,6 +33,7 @@ export default function AddEvent(props: {
className?: string; className?: string;
isOpen: boolean; isOpen: boolean;
onOpenChange: (isOpen: boolean) => void; onOpenChange: (isOpen: boolean) => void;
onSuccess?: () => void;
}) { }) {
// initial state for the inputs // initial state for the inputs
const initialState: state = { const initialState: state = {
@@ -73,6 +74,8 @@ export default function AddEvent(props: {
zustand.getState().setEvents(await result.json()); zustand.getState().setEvents(await result.json());
props.onOpenChange(false); props.onOpenChange(false);
props.onSuccess?.();
} }
} }

View File

@@ -84,8 +84,10 @@ export class DateFormatter {
export function vaidatePassword(password: string): string[] { export function vaidatePassword(password: string): string[] {
const errors = []; const errors = [];
if (password.length < 1) { if (password.length < 12) {
errors.push("Password must be 16 characters or more"); errors.push("Password must be 12 characters or more");
} else if (password.length > 64) {
errors.push("Password must be 64 characters or short");
} }
return errors; return errors;

View File

@@ -20,7 +20,20 @@ export default {
theme: { theme: {
extend: { extend: {
colors: { colors: {
highlight: HIGHLIGHT, highlight: {
DEFAULT: HIGHLIGHT,
"50": "hsl(0,85.7%,97.3%)",
"100": "hsl(0,93.3%,94.1%)",
"200": "hsl(0,96.3%,89.4%)",
"300": "hsl(0,93.5%,81.8%)",
"400": "hsl(0,90.6%,70.8%)",
"500": "hsl(0,84.2%,60.2%)",
"600": "hsl(0,72.2%,50.6%)",
"700": "hsl(0,73.7%,41.8%)",
"800": "hsl(0,70%,35.3%)",
"900": "hsl(0,62.8%,30.6%)",
"950": "hsl(0,74.7%,15.5%)",
},
foreground: FOREGROUND, foreground: FOREGROUND,
"accent-1": ACCENT1, "accent-1": ACCENT1,
"accent-2": ACCENT2, "accent-2": ACCENT2,