forked from mirrors/homebox
bd321af29f
* new PR tasks * add homebox to know words * formatting * bump deps * generate db models * ts errors * drop id * fix accessor * drop unused time field * change CI * add expected error * add type check * resolve serveral type errors * hoise in CI
236 lines
6.5 KiB
Go
236 lines
6.5 KiB
Go
package services
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"time"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/hay-kot/homebox/backend/internal/data/ent/authroles"
|
|
"github.com/hay-kot/homebox/backend/internal/data/repo"
|
|
"github.com/hay-kot/homebox/backend/pkgs/hasher"
|
|
"github.com/rs/zerolog/log"
|
|
)
|
|
|
|
var (
|
|
oneWeek = time.Hour * 24 * 7
|
|
ErrorInvalidLogin = errors.New("invalid username or password")
|
|
ErrorInvalidToken = errors.New("invalid token")
|
|
ErrorTokenIdMismatch = errors.New("token id mismatch")
|
|
)
|
|
|
|
type UserService struct {
|
|
repos *repo.AllRepos
|
|
}
|
|
|
|
type (
|
|
UserRegistration struct {
|
|
GroupToken string `json:"token"`
|
|
Name string `json:"name"`
|
|
Email string `json:"email"`
|
|
Password string `json:"password"`
|
|
}
|
|
UserAuthTokenDetail struct {
|
|
Raw string `json:"raw"`
|
|
AttachmentToken string `json:"attachmentToken"`
|
|
ExpiresAt time.Time `json:"expiresAt"`
|
|
}
|
|
LoginForm struct {
|
|
Username string `json:"username"`
|
|
Password string `json:"password"`
|
|
}
|
|
)
|
|
|
|
// RegisterUser creates a new user and group in the data with the provided data. It also bootstraps the user's group
|
|
// with default Labels and Locations.
|
|
func (svc *UserService) RegisterUser(ctx context.Context, data UserRegistration) (repo.UserOut, error) {
|
|
log.Debug().
|
|
Str("name", data.Name).
|
|
Str("email", data.Email).
|
|
Str("groupToken", data.GroupToken).
|
|
Msg("Registering new user")
|
|
|
|
var (
|
|
err error
|
|
group repo.Group
|
|
token repo.GroupInvitation
|
|
|
|
// creatingGroup is true if the user is creating a new group.
|
|
creatingGroup = false
|
|
)
|
|
|
|
switch data.GroupToken {
|
|
case "":
|
|
creatingGroup = true
|
|
group, err = svc.repos.Groups.GroupCreate(ctx, "Home")
|
|
if err != nil {
|
|
log.Err(err).Msg("Failed to create group")
|
|
return repo.UserOut{}, err
|
|
}
|
|
default:
|
|
token, err = svc.repos.Groups.InvitationGet(ctx, hasher.HashToken(data.GroupToken))
|
|
if err != nil {
|
|
log.Err(err).Msg("Failed to get invitation token")
|
|
return repo.UserOut{}, err
|
|
}
|
|
group = token.Group
|
|
}
|
|
|
|
hashed, _ := hasher.HashPassword(data.Password)
|
|
usrCreate := repo.UserCreate{
|
|
Name: data.Name,
|
|
Email: data.Email,
|
|
Password: hashed,
|
|
IsSuperuser: false,
|
|
GroupID: group.ID,
|
|
IsOwner: creatingGroup,
|
|
}
|
|
|
|
usr, err := svc.repos.Users.Create(ctx, usrCreate)
|
|
if err != nil {
|
|
return repo.UserOut{}, err
|
|
}
|
|
|
|
// Create the default labels and locations for the group.
|
|
if creatingGroup {
|
|
for _, label := range defaultLabels() {
|
|
_, err := svc.repos.Labels.Create(ctx, group.ID, label)
|
|
if err != nil {
|
|
return repo.UserOut{}, err
|
|
}
|
|
}
|
|
|
|
for _, location := range defaultLocations() {
|
|
_, err := svc.repos.Locations.Create(ctx, group.ID, location)
|
|
if err != nil {
|
|
return repo.UserOut{}, err
|
|
}
|
|
}
|
|
}
|
|
|
|
// Decrement the invitation token if it was used.
|
|
if token.ID != uuid.Nil {
|
|
err = svc.repos.Groups.InvitationUpdate(ctx, token.ID, token.Uses-1)
|
|
if err != nil {
|
|
log.Err(err).Msg("Failed to update invitation token")
|
|
return repo.UserOut{}, err
|
|
}
|
|
}
|
|
|
|
return usr, nil
|
|
}
|
|
|
|
// GetSelf returns the user that is currently logged in based of the token provided within
|
|
func (svc *UserService) GetSelf(ctx context.Context, requestToken string) (repo.UserOut, error) {
|
|
hash := hasher.HashToken(requestToken)
|
|
return svc.repos.AuthTokens.GetUserFromToken(ctx, hash)
|
|
}
|
|
|
|
func (svc *UserService) UpdateSelf(ctx context.Context, ID uuid.UUID, data repo.UserUpdate) (repo.UserOut, error) {
|
|
err := svc.repos.Users.Update(ctx, ID, data)
|
|
if err != nil {
|
|
return repo.UserOut{}, err
|
|
}
|
|
|
|
return svc.repos.Users.GetOneId(ctx, ID)
|
|
}
|
|
|
|
// ============================================================================
|
|
// User Authentication
|
|
|
|
func (svc *UserService) createSessionToken(ctx context.Context, userId uuid.UUID) (UserAuthTokenDetail, error) {
|
|
attachmentToken := hasher.GenerateToken()
|
|
attachmentData := repo.UserAuthTokenCreate{
|
|
UserID: userId,
|
|
TokenHash: attachmentToken.Hash,
|
|
ExpiresAt: time.Now().Add(oneWeek),
|
|
}
|
|
|
|
_, err := svc.repos.AuthTokens.CreateToken(ctx, attachmentData, authroles.RoleAttachments)
|
|
if err != nil {
|
|
return UserAuthTokenDetail{}, err
|
|
}
|
|
|
|
userToken := hasher.GenerateToken()
|
|
data := repo.UserAuthTokenCreate{
|
|
UserID: userId,
|
|
TokenHash: userToken.Hash,
|
|
ExpiresAt: time.Now().Add(oneWeek),
|
|
}
|
|
|
|
created, err := svc.repos.AuthTokens.CreateToken(ctx, data, authroles.RoleUser)
|
|
if err != nil {
|
|
return UserAuthTokenDetail{}, err
|
|
}
|
|
|
|
return UserAuthTokenDetail{
|
|
Raw: userToken.Raw,
|
|
ExpiresAt: created.ExpiresAt,
|
|
AttachmentToken: attachmentToken.Raw,
|
|
}, nil
|
|
}
|
|
|
|
func (svc *UserService) Login(ctx context.Context, username, password string) (UserAuthTokenDetail, error) {
|
|
usr, err := svc.repos.Users.GetOneEmail(ctx, username)
|
|
if err != nil {
|
|
// SECURITY: Perform hash to ensure response times are the same
|
|
hasher.CheckPasswordHash("not-a-real-password", "not-a-real-password")
|
|
return UserAuthTokenDetail{}, ErrorInvalidLogin
|
|
}
|
|
|
|
if !hasher.CheckPasswordHash(password, usr.PasswordHash) {
|
|
return UserAuthTokenDetail{}, ErrorInvalidLogin
|
|
}
|
|
|
|
return svc.createSessionToken(ctx, usr.ID)
|
|
}
|
|
|
|
func (svc *UserService) Logout(ctx context.Context, token string) error {
|
|
hash := hasher.HashToken(token)
|
|
err := svc.repos.AuthTokens.DeleteToken(ctx, hash)
|
|
return err
|
|
}
|
|
|
|
func (svc *UserService) RenewToken(ctx context.Context, token string) (UserAuthTokenDetail, error) {
|
|
hash := hasher.HashToken(token)
|
|
|
|
dbToken, err := svc.repos.AuthTokens.GetUserFromToken(ctx, hash)
|
|
if err != nil {
|
|
return UserAuthTokenDetail{}, ErrorInvalidToken
|
|
}
|
|
|
|
return svc.createSessionToken(ctx, dbToken.ID)
|
|
}
|
|
|
|
// DeleteSelf deletes the user that is currently logged based of the provided UUID
|
|
// There is _NO_ protection against deleting the wrong user, as such this should only
|
|
// be used when the identify of the user has been confirmed.
|
|
func (svc *UserService) DeleteSelf(ctx context.Context, ID uuid.UUID) error {
|
|
return svc.repos.Users.Delete(ctx, ID)
|
|
}
|
|
|
|
func (svc *UserService) ChangePassword(ctx Context, current string, new string) (ok bool) {
|
|
usr, err := svc.repos.Users.GetOneId(ctx, ctx.UID)
|
|
if err != nil {
|
|
return false
|
|
}
|
|
|
|
if !hasher.CheckPasswordHash(current, usr.PasswordHash) {
|
|
log.Err(errors.New("current password is incorrect")).Msg("Failed to change password")
|
|
return false
|
|
}
|
|
|
|
hashed, err := hasher.HashPassword(new)
|
|
if err != nil {
|
|
log.Err(err).Msg("Failed to hash password")
|
|
return false
|
|
}
|
|
|
|
err = svc.repos.Users.ChangePassword(ctx.Context, ctx.UID, hashed)
|
|
if err != nil {
|
|
log.Err(err).Msg("Failed to change password")
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|