2019-01-05 22:44:33 +00:00
|
|
|
package users
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/filebrowser/filebrowser/v2/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
// StorageBackend is the interface to implement for a users storage.
|
|
|
|
type StorageBackend interface {
|
2019-01-08 02:03:52 +00:00
|
|
|
GetBy(interface{}) (*User, error)
|
2019-01-05 22:44:33 +00:00
|
|
|
Gets() ([]*User, error)
|
|
|
|
Save(u *User) error
|
|
|
|
Update(u *User, fields ...string) error
|
|
|
|
DeleteByID(uint) error
|
|
|
|
DeleteByUsername(string) error
|
|
|
|
}
|
|
|
|
|
|
|
|
// Storage is a users storage.
|
|
|
|
type Storage struct {
|
|
|
|
back StorageBackend
|
|
|
|
updated map[uint]int64
|
|
|
|
mux sync.RWMutex
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewStorage creates a users storage from a backend.
|
|
|
|
func NewStorage(back StorageBackend) *Storage {
|
|
|
|
return &Storage{
|
|
|
|
back: back,
|
|
|
|
updated: map[uint]int64{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get allows you to get a user by its name or username. The provided
|
|
|
|
// id must be a string for username lookup or a uint for id lookup. If id
|
|
|
|
// is neither, a ErrInvalidDataType will be returned.
|
2019-01-08 02:03:52 +00:00
|
|
|
func (s *Storage) Get(baseScope string, id interface{}) (user *User, err error) {
|
|
|
|
user, err = s.back.GetBy(id)
|
2019-01-05 22:44:33 +00:00
|
|
|
if err != nil {
|
2019-01-08 02:03:52 +00:00
|
|
|
return
|
2019-01-05 22:44:33 +00:00
|
|
|
}
|
2020-05-31 23:12:36 +00:00
|
|
|
if err := user.Clean(baseScope); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-01-08 02:03:52 +00:00
|
|
|
return
|
2019-01-05 22:44:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Gets gets a list of all users.
|
2019-01-06 13:01:42 +00:00
|
|
|
func (s *Storage) Gets(baseScope string) ([]*User, error) {
|
2019-01-05 22:44:33 +00:00
|
|
|
users, err := s.back.Gets()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, user := range users {
|
2020-05-31 23:12:36 +00:00
|
|
|
if err := user.Clean(baseScope); err != nil { //nolint:shadow
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-01-05 22:44:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return users, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update updates a user in the database.
|
|
|
|
func (s *Storage) Update(user *User, fields ...string) error {
|
2019-01-06 13:01:42 +00:00
|
|
|
err := user.Clean("", fields...)
|
2019-01-05 22:44:33 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = s.back.Update(user, fields...)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
s.mux.Lock()
|
|
|
|
s.updated[user.ID] = time.Now().Unix()
|
|
|
|
s.mux.Unlock()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save saves the user in a storage.
|
|
|
|
func (s *Storage) Save(user *User) error {
|
2019-01-06 13:01:42 +00:00
|
|
|
if err := user.Clean(""); err != nil {
|
2019-01-05 22:44:33 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.back.Save(user)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete allows you to delete a user by its name or username. The provided
|
|
|
|
// id must be a string for username lookup or a uint for id lookup. If id
|
|
|
|
// is neither, a ErrInvalidDataType will be returned.
|
|
|
|
func (s *Storage) Delete(id interface{}) (err error) {
|
2019-01-23 15:58:06 +00:00
|
|
|
switch id := id.(type) {
|
2019-01-05 22:44:33 +00:00
|
|
|
case string:
|
2019-01-23 15:58:06 +00:00
|
|
|
err = s.back.DeleteByUsername(id)
|
2019-01-05 22:44:33 +00:00
|
|
|
case uint:
|
2019-01-23 15:58:06 +00:00
|
|
|
err = s.back.DeleteByID(id)
|
2019-01-05 22:44:33 +00:00
|
|
|
default:
|
|
|
|
err = errors.ErrInvalidDataType
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// LastUpdate gets the timestamp for the last update of an user.
|
|
|
|
func (s *Storage) LastUpdate(id uint) int64 {
|
|
|
|
s.mux.RLock()
|
|
|
|
defer s.mux.RUnlock()
|
|
|
|
if val, ok := s.updated[id]; ok {
|
|
|
|
return val
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|