196 lines
4.7 KiB
Go
196 lines
4.7 KiB
Go
![]() |
package dynamodb
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"errors"
|
||
|
"time"
|
||
|
|
||
|
"github.com/authorizerdev/authorizer/server/constants"
|
||
|
"github.com/authorizerdev/authorizer/server/db/models"
|
||
|
"github.com/authorizerdev/authorizer/server/graph/model"
|
||
|
"github.com/authorizerdev/authorizer/server/memorystore"
|
||
|
"github.com/google/uuid"
|
||
|
"github.com/guregu/dynamo"
|
||
|
log "github.com/sirupsen/logrus"
|
||
|
)
|
||
|
|
||
|
// AddUser to save user information in database
|
||
|
func (p *provider) AddUser(ctx context.Context, user models.User) (models.User, error) {
|
||
|
collection := p.db.Table(models.Collections.User)
|
||
|
|
||
|
if user.ID == "" {
|
||
|
user.ID = uuid.New().String()
|
||
|
}
|
||
|
|
||
|
if user.Roles == "" {
|
||
|
defaultRoles, err := memorystore.Provider.GetStringStoreEnvVariable(constants.EnvKeyDefaultRoles)
|
||
|
if err != nil {
|
||
|
return user, err
|
||
|
}
|
||
|
user.Roles = defaultRoles
|
||
|
}
|
||
|
|
||
|
user.CreatedAt = time.Now().Unix()
|
||
|
user.UpdatedAt = time.Now().Unix()
|
||
|
|
||
|
err := collection.Put(user).RunWithContext(ctx)
|
||
|
|
||
|
if err != nil {
|
||
|
return user, err
|
||
|
}
|
||
|
return user, nil
|
||
|
}
|
||
|
|
||
|
// UpdateUser to update user information in database
|
||
|
func (p *provider) UpdateUser(ctx context.Context, user models.User) (models.User, error) {
|
||
|
collection := p.db.Table(models.Collections.User)
|
||
|
|
||
|
if user.ID != "" {
|
||
|
|
||
|
user.UpdatedAt = time.Now().Unix()
|
||
|
|
||
|
err := UpdateByHashKey(collection, "id", user.ID, user)
|
||
|
if err != nil {
|
||
|
return user, err
|
||
|
}
|
||
|
|
||
|
if err != nil {
|
||
|
return user, err
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return user, nil
|
||
|
}
|
||
|
|
||
|
// DeleteUser to delete user information from database
|
||
|
func (p *provider) DeleteUser(ctx context.Context, user models.User) error {
|
||
|
collection := p.db.Table(models.Collections.User)
|
||
|
sessionCollection := p.db.Table(models.Collections.Session)
|
||
|
|
||
|
if user.ID != "" {
|
||
|
err := collection.Delete("id", user.ID).Run()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
_, err = sessionCollection.Batch("id").Write().Delete(dynamo.Keys{"user_id", user.ID}).RunWithContext(ctx)
|
||
|
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// ListUsers to get list of users from database
|
||
|
func (p *provider) ListUsers(ctx context.Context, pagination model.Pagination) (*model.Users, error) {
|
||
|
var user models.User
|
||
|
var lastEval dynamo.PagingKey
|
||
|
var iter dynamo.PagingIter
|
||
|
var iteration int64 = 0
|
||
|
|
||
|
collection := p.db.Table(models.Collections.User)
|
||
|
users := []*model.User{}
|
||
|
|
||
|
paginationClone := pagination
|
||
|
scanner := collection.Scan()
|
||
|
count, err := scanner.Count()
|
||
|
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
for (paginationClone.Offset + paginationClone.Limit) > iteration {
|
||
|
iter = scanner.StartFrom(lastEval).Limit(paginationClone.Limit).Iter()
|
||
|
for iter.NextWithContext(ctx, &user) {
|
||
|
if paginationClone.Offset == iteration {
|
||
|
users = append(users, user.AsAPIUser())
|
||
|
}
|
||
|
}
|
||
|
lastEval = iter.LastEvaluatedKey()
|
||
|
iteration += paginationClone.Limit
|
||
|
}
|
||
|
|
||
|
err = iter.Err()
|
||
|
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
paginationClone.Total = count
|
||
|
|
||
|
return &model.Users{
|
||
|
Pagination: &paginationClone,
|
||
|
Users: users,
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
// GetUserByEmail to get user information from database using email address
|
||
|
func (p *provider) GetUserByEmail(ctx context.Context, email string) (models.User, error) {
|
||
|
var users []models.User
|
||
|
var user models.User
|
||
|
|
||
|
collection := p.db.Table(models.Collections.User)
|
||
|
err := collection.Scan().Filter("'email' = ?", email).AllWithContext(ctx, &users)
|
||
|
|
||
|
if err != nil {
|
||
|
return user, nil
|
||
|
}
|
||
|
|
||
|
if len(users) > 0 {
|
||
|
user = users[0]
|
||
|
return user, nil
|
||
|
} else {
|
||
|
return user, errors.New("no record found")
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
// GetUserByID to get user information from database using user ID
|
||
|
func (p *provider) GetUserByID(ctx context.Context, id string) (models.User, error) {
|
||
|
collection := p.db.Table(models.Collections.User)
|
||
|
var user models.User
|
||
|
err := collection.Get("id", id).OneWithContext(ctx, &user)
|
||
|
|
||
|
if err != nil {
|
||
|
if user.Email == "" {
|
||
|
return user, errors.New("no documets found")
|
||
|
} else {
|
||
|
return user, nil
|
||
|
}
|
||
|
}
|
||
|
return user, nil
|
||
|
}
|
||
|
|
||
|
// UpdateUsers to update multiple users, with parameters of user IDs slice
|
||
|
// If ids set to nil / empty all the users will be updated
|
||
|
func (p *provider) UpdateUsers(ctx context.Context, data map[string]interface{}, ids []string) error {
|
||
|
// set updated_at time for all users
|
||
|
userCollection := p.db.Table(models.Collections.User)
|
||
|
var allUsers []models.User
|
||
|
var res int64 = 0
|
||
|
var err error
|
||
|
if len(ids) > 0 {
|
||
|
for _, v := range ids {
|
||
|
err = UpdateByHashKey(userCollection, "id", v, data)
|
||
|
}
|
||
|
} else {
|
||
|
// as there is no facility to update all doc - https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/SQLtoNoSQL.UpdateData.html
|
||
|
userCollection.Scan().All(&allUsers)
|
||
|
|
||
|
for _, user := range allUsers {
|
||
|
err = UpdateByHashKey(userCollection, "id", user.ID, data)
|
||
|
if err != nil {
|
||
|
res = res + 1
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if err != nil {
|
||
|
return err
|
||
|
} else {
|
||
|
log.Info("Updated users: ", res)
|
||
|
}
|
||
|
return nil
|
||
|
}
|