2024-03-02 02:38:26 +01:00
|
|
|
package handlers
|
|
|
|
|
|
|
|
import (
|
2024-03-15 15:28:45 +01:00
|
|
|
"strconv"
|
2024-03-06 12:51:46 +01:00
|
|
|
"time"
|
2024-03-02 02:38:26 +01:00
|
|
|
"ttime/internal/database"
|
|
|
|
"ttime/internal/types"
|
|
|
|
|
|
|
|
"github.com/gofiber/fiber/v2"
|
2024-03-06 12:51:46 +01:00
|
|
|
"github.com/golang-jwt/jwt/v5"
|
2024-03-02 02:38:26 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// The actual interface that we will use
|
|
|
|
type GlobalState interface {
|
2024-03-14 21:25:14 +01:00
|
|
|
Register(c *fiber.Ctx) error // To register a new user
|
|
|
|
UserDelete(c *fiber.Ctx) error // To delete a user
|
|
|
|
Login(c *fiber.Ctx) error // To get the token
|
|
|
|
LoginRenew(c *fiber.Ctx) error // To renew the token
|
|
|
|
CreateProject(c *fiber.Ctx) error // To create a new project
|
|
|
|
GetUserProjects(c *fiber.Ctx) error // To get all projects
|
2024-03-16 22:47:19 +01:00
|
|
|
SubmitWeeklyReport(c *fiber.Ctx) error
|
2024-03-06 10:14:54 +01:00
|
|
|
// GetProject(c *fiber.Ctx) error // To get a specific project
|
|
|
|
// UpdateProject(c *fiber.Ctx) error // To update a project
|
|
|
|
// DeleteProject(c *fiber.Ctx) error // To delete a project
|
|
|
|
// CreateTask(c *fiber.Ctx) error // To create a new task
|
|
|
|
// GetTasks(c *fiber.Ctx) error // To get all tasks
|
|
|
|
// GetTask(c *fiber.Ctx) error // To get a specific task
|
|
|
|
// UpdateTask(c *fiber.Ctx) error // To update a task
|
|
|
|
// DeleteTask(c *fiber.Ctx) error // To delete a task
|
|
|
|
// CreateCollection(c *fiber.Ctx) error // To create a new collection
|
|
|
|
// GetCollections(c *fiber.Ctx) error // To get all collections
|
|
|
|
// GetCollection(c *fiber.Ctx) error // To get a specific collection
|
|
|
|
// UpdateCollection(c *fiber.Ctx) error // To update a collection
|
|
|
|
// DeleteCollection(c *fiber.Ctx) error // To delete a collection
|
|
|
|
// SignCollection(c *fiber.Ctx) error // To sign a collection
|
|
|
|
GetButtonCount(c *fiber.Ctx) error // For demonstration purposes
|
|
|
|
IncrementButtonCount(c *fiber.Ctx) error // For demonstration purposes
|
2024-03-14 22:48:06 +01:00
|
|
|
ListAllUsers(c *fiber.Ctx) error // To get a list of all users in the application database
|
2024-03-14 22:56:50 +01:00
|
|
|
ListAllUsersProject(c *fiber.Ctx) error // To get a list of all users for a specific project
|
2024-03-14 23:06:10 +01:00
|
|
|
ProjectRoleChange(c *fiber.Ctx) error // To change a users role in a project
|
2024-03-02 02:38:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// "Constructor"
|
|
|
|
func NewGlobalState(db database.Database) GlobalState {
|
2024-03-06 09:41:36 +01:00
|
|
|
return &GState{Db: db, ButtonCount: 0}
|
2024-03-02 02:38:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// The global state, which implements all the handlers
|
|
|
|
type GState struct {
|
2024-03-06 09:41:36 +01:00
|
|
|
Db database.Database
|
|
|
|
ButtonCount int
|
2024-03-02 02:38:26 +01:00
|
|
|
}
|
|
|
|
|
2024-03-08 10:16:56 +01:00
|
|
|
// Register is a simple handler that registers a new user
|
|
|
|
//
|
|
|
|
// @Summary Register a new user
|
|
|
|
// @Description Register a new user
|
|
|
|
// @Tags User
|
|
|
|
// @Accept json
|
|
|
|
// @Produce json
|
2024-03-08 10:30:17 +01:00
|
|
|
// @Success 200 {string} string "User added"
|
|
|
|
// @Failure 400 {string} string "Bad request"
|
|
|
|
// @Failure 500 {string} string "Internal server error"
|
2024-03-08 10:16:56 +01:00
|
|
|
// @Router /api/register [post]
|
2024-03-02 02:38:26 +01:00
|
|
|
func (gs *GState) Register(c *fiber.Ctx) error {
|
2024-03-12 20:44:40 +01:00
|
|
|
u := new(types.NewUser)
|
2024-03-02 02:38:26 +01:00
|
|
|
if err := c.BodyParser(u); err != nil {
|
|
|
|
return c.Status(400).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := gs.Db.AddUser(u.Username, u.Password); err != nil {
|
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.Status(200).SendString("User added")
|
|
|
|
}
|
2024-03-06 09:41:36 +01:00
|
|
|
|
2024-03-06 10:14:54 +01:00
|
|
|
// This path should obviously be protected in the future
|
|
|
|
// UserDelete deletes a user from the database
|
|
|
|
func (gs *GState) UserDelete(c *fiber.Ctx) error {
|
2024-03-16 22:47:19 +01:00
|
|
|
// Read from path parameters
|
|
|
|
username := c.Params("username")
|
|
|
|
|
|
|
|
// Read username from Locals
|
|
|
|
auth_username := c.Locals("user").(*jwt.Token).Claims.(jwt.MapClaims)["name"].(string)
|
|
|
|
|
|
|
|
if username != auth_username {
|
|
|
|
return c.Status(403).SendString("You can only delete yourself")
|
2024-03-06 10:14:54 +01:00
|
|
|
}
|
|
|
|
|
2024-03-16 22:47:19 +01:00
|
|
|
if err := gs.Db.RemoveUser(username); err != nil {
|
2024-03-06 10:14:54 +01:00
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.Status(200).SendString("User deleted")
|
|
|
|
}
|
|
|
|
|
2024-03-06 09:41:36 +01:00
|
|
|
func (gs *GState) GetButtonCount(c *fiber.Ctx) error {
|
|
|
|
return c.Status(200).JSON(fiber.Map{"pressCount": gs.ButtonCount})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gs *GState) IncrementButtonCount(c *fiber.Ctx) error {
|
|
|
|
gs.ButtonCount++
|
|
|
|
return c.Status(200).JSON(fiber.Map{"pressCount": gs.ButtonCount})
|
|
|
|
}
|
2024-03-06 12:51:46 +01:00
|
|
|
|
|
|
|
// Login is a simple login handler that returns a JWT token
|
|
|
|
func (gs *GState) Login(c *fiber.Ctx) error {
|
2024-03-17 03:39:31 +01:00
|
|
|
// The body type is identical to a NewUser
|
|
|
|
u := new(types.NewUser)
|
|
|
|
if err := c.BodyParser(u); err != nil {
|
|
|
|
return c.Status(400).SendString(err.Error())
|
|
|
|
}
|
2024-03-06 12:51:46 +01:00
|
|
|
|
2024-03-17 03:39:31 +01:00
|
|
|
if !gs.Db.CheckUser(u.Username, u.Password) {
|
|
|
|
println("User not found")
|
2024-03-06 12:51:46 +01:00
|
|
|
return c.SendStatus(fiber.StatusUnauthorized)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the Claims
|
|
|
|
claims := jwt.MapClaims{
|
2024-03-17 03:39:31 +01:00
|
|
|
"name": u.Username,
|
2024-03-06 12:51:46 +01:00
|
|
|
"admin": false,
|
|
|
|
"exp": time.Now().Add(time.Hour * 72).Unix(),
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create token
|
|
|
|
token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)
|
|
|
|
|
|
|
|
// Generate encoded token and send it as response.
|
|
|
|
t, err := token.SignedString([]byte("secret"))
|
|
|
|
if err != nil {
|
|
|
|
return c.SendStatus(fiber.StatusInternalServerError)
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.JSON(fiber.Map{"token": t})
|
|
|
|
}
|
|
|
|
|
|
|
|
// LoginRenew is a simple handler that renews the token
|
|
|
|
func (gs *GState) LoginRenew(c *fiber.Ctx) error {
|
2024-03-07 11:33:06 +01:00
|
|
|
// For testing: curl localhost:3000/restricted -H "Authorization: Bearer <token>"
|
2024-03-06 12:51:46 +01:00
|
|
|
user := c.Locals("user").(*jwt.Token)
|
|
|
|
claims := user.Claims.(jwt.MapClaims)
|
|
|
|
claims["exp"] = time.Now().Add(time.Hour * 72).Unix()
|
|
|
|
renewed := jwt.MapClaims{
|
|
|
|
"name": claims["name"],
|
|
|
|
"admin": claims["admin"],
|
|
|
|
"exp": claims["exp"],
|
|
|
|
}
|
|
|
|
token := jwt.NewWithClaims(jwt.SigningMethodHS256, renewed)
|
|
|
|
t, err := token.SignedString([]byte("secret"))
|
|
|
|
if err != nil {
|
|
|
|
return c.SendStatus(fiber.StatusInternalServerError)
|
|
|
|
}
|
|
|
|
return c.JSON(fiber.Map{"token": t})
|
|
|
|
}
|
2024-03-12 20:44:40 +01:00
|
|
|
|
|
|
|
// CreateProject is a simple handler that creates a new project
|
|
|
|
func (gs *GState) CreateProject(c *fiber.Ctx) error {
|
|
|
|
user := c.Locals("user").(*jwt.Token)
|
|
|
|
|
|
|
|
p := new(types.NewProject)
|
|
|
|
if err := c.BodyParser(p); err != nil {
|
|
|
|
return c.Status(400).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the username from the token and set it as the owner of the project
|
|
|
|
// This is ugly but
|
|
|
|
claims := user.Claims.(jwt.MapClaims)
|
2024-03-16 22:57:48 +01:00
|
|
|
owner := claims["name"].(string)
|
2024-03-12 20:44:40 +01:00
|
|
|
|
2024-03-16 22:57:48 +01:00
|
|
|
if err := gs.Db.AddProject(p.Name, p.Description, owner); err != nil {
|
2024-03-12 20:44:40 +01:00
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.Status(200).SendString("Project added")
|
|
|
|
}
|
2024-03-14 21:25:14 +01:00
|
|
|
|
|
|
|
// GetUserProjects returns all projects that the user is a member of
|
|
|
|
func (gs *GState) GetUserProjects(c *fiber.Ctx) error {
|
|
|
|
// First we get the username from the token
|
|
|
|
user := c.Locals("user").(*jwt.Token)
|
|
|
|
claims := user.Claims.(jwt.MapClaims)
|
|
|
|
username := claims["name"].(string)
|
|
|
|
|
|
|
|
// Then dip into the database to get the projects
|
|
|
|
projects, err := gs.Db.GetProjectsForUser(username)
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return a json serialized list of projects
|
|
|
|
return c.JSON(projects)
|
|
|
|
}
|
2024-03-14 22:48:06 +01:00
|
|
|
|
2024-03-14 22:56:50 +01:00
|
|
|
// ListAllUsers is a handler that returns a list of all users in the application database
|
2024-03-14 22:48:06 +01:00
|
|
|
func (gs *GState) ListAllUsers(c *fiber.Ctx) error {
|
|
|
|
// Get all users from the database
|
|
|
|
users, err := gs.Db.GetAllUsersApplication()
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the list of users as JSON
|
|
|
|
return c.JSON(users)
|
|
|
|
}
|
2024-03-14 22:56:50 +01:00
|
|
|
|
|
|
|
func (gs *GState) ListAllUsersProject(c *fiber.Ctx) error {
|
|
|
|
// Extract the project name from the request parameters or body
|
|
|
|
projectName := c.Params("projectName")
|
|
|
|
|
|
|
|
// Get all users associated with the project from the database
|
|
|
|
users, err := gs.Db.GetAllUsersProject(projectName)
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the list of users as JSON
|
|
|
|
return c.JSON(users)
|
|
|
|
}
|
2024-03-14 23:06:10 +01:00
|
|
|
|
|
|
|
// ProjectRoleChange is a handler that changes a user's role within a project
|
|
|
|
func (gs *GState) ProjectRoleChange(c *fiber.Ctx) error {
|
|
|
|
// Extract the necessary parameters from the request
|
|
|
|
username := c.Params("username")
|
|
|
|
projectName := c.Params("projectName")
|
|
|
|
role := c.Params("role")
|
|
|
|
|
|
|
|
// Change the user's role within the project in the database
|
|
|
|
if err := gs.Db.ChangeUserRole(username, projectName, role); err != nil {
|
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return a success message
|
|
|
|
return c.SendStatus(fiber.StatusOK)
|
|
|
|
}
|
2024-03-15 15:28:45 +01:00
|
|
|
|
|
|
|
// GetProject retrieves a specific project by its ID
|
|
|
|
func (gs *GState) GetProject(c *fiber.Ctx) error {
|
|
|
|
// Extract the project ID from the request parameters or body
|
|
|
|
projectID := c.Params("projectID")
|
|
|
|
|
|
|
|
// Parse the project ID into an integer
|
|
|
|
projectIDInt, err := strconv.Atoi(projectID)
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(400).SendString("Invalid project ID")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the project from the database by its ID
|
|
|
|
project, err := gs.Db.GetProject(projectIDInt)
|
|
|
|
if err != nil {
|
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the project as JSON
|
|
|
|
return c.JSON(project)
|
|
|
|
}
|
2024-03-16 22:47:19 +01:00
|
|
|
|
|
|
|
func (gs *GState) SubmitWeeklyReport(c *fiber.Ctx) error {
|
|
|
|
// Extract the necessary parameters from the token
|
|
|
|
user := c.Locals("user").(*jwt.Token)
|
|
|
|
claims := user.Claims.(jwt.MapClaims)
|
|
|
|
username := claims["name"].(string)
|
|
|
|
|
|
|
|
report := new(types.NewWeeklyReport)
|
|
|
|
if err := c.BodyParser(report); err != nil {
|
|
|
|
return c.Status(400).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
2024-03-17 04:16:26 +01:00
|
|
|
// Make sure all the fields of the report are valid
|
|
|
|
if report.Week < 1 || report.Week > 52 {
|
|
|
|
return c.Status(400).SendString("Invalid week number")
|
|
|
|
}
|
|
|
|
if report.DevelopmentTime < 0 || report.MeetingTime < 0 || report.AdminTime < 0 || report.OwnWorkTime < 0 || report.StudyTime < 0 || report.TestingTime < 0 {
|
|
|
|
return c.Status(400).SendString("Invalid time report")
|
|
|
|
}
|
|
|
|
|
2024-03-16 22:47:19 +01:00
|
|
|
if err := gs.Db.AddWeeklyReport(report.ProjectName, username, report.Week, report.DevelopmentTime, report.MeetingTime, report.AdminTime, report.OwnWorkTime, report.StudyTime, report.TestingTime); err != nil {
|
|
|
|
return c.Status(500).SendString(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.Status(200).SendString("Time report added")
|
|
|
|
}
|