functionally completed user editing
This commit is contained in:
@@ -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
|
||||||
|
|||||||
@@ -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 SetAdmin(userName string, admin bool) error {
|
||||||
}
|
_, err := db.DB.Exec("UPDATE USERS SET admin = ? WHERE name = ?", admin, userName)
|
||||||
}
|
|
||||||
|
|
||||||
func init() {
|
return err
|
||||||
c = cache.New(24 * time.Hour)
|
|
||||||
|
|
||||||
c.OnExpired(refresh)
|
|
||||||
|
|
||||||
refresh()
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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*",
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -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)),
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -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();
|
||||||
|
|
||||||
|
|||||||
@@ -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:
|
||||||
|
|||||||
@@ -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"
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
137
client/src/app/admin/EditUser.tsx
Normal file
137
client/src/app/admin/EditUser.tsx
Normal 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>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -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() {
|
||||||
|
const result = await apiCall("GET", "users");
|
||||||
|
|
||||||
|
if (result.ok) {
|
||||||
|
const users = (await result.json()) as User[];
|
||||||
|
|
||||||
return {
|
return {
|
||||||
items: [
|
items: users,
|
||||||
{ userName: "admin", admin: true },
|
|
||||||
{ userName: "foo", admin: false },
|
|
||||||
{ userName: "bar", admin: true },
|
|
||||||
],
|
|
||||||
};
|
};
|
||||||
|
} 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>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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) {
|
||||||
|
|||||||
@@ -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 {
|
||||||
|
|||||||
@@ -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?.();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
Reference in New Issue
Block a user