2019-04-24 21:52:08 +00:00
|
|
|
package authentication
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
2019-12-27 16:55:00 +00:00
|
|
|
"strings"
|
2019-04-24 21:52:08 +00:00
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/asaskevich/govalidator"
|
|
|
|
|
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
// FileUserProvider is a provider reading details from a file.
|
|
|
|
type FileUserProvider struct {
|
|
|
|
path *string
|
|
|
|
database *DatabaseModel
|
|
|
|
lock *sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
// UserDetailsModel is the model of user details in the file database.
|
|
|
|
type UserDetailsModel struct {
|
|
|
|
HashedPassword string `yaml:"password" valid:"required"`
|
|
|
|
Email string `yaml:"email"`
|
|
|
|
Groups []string `yaml:"groups"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// DatabaseModel is the model of users file database.
|
|
|
|
type DatabaseModel struct {
|
|
|
|
Users map[string]UserDetailsModel `yaml:"users" valid:"required"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewFileUserProvider creates a new instance of FileUserProvider.
|
|
|
|
func NewFileUserProvider(filepath string) *FileUserProvider {
|
|
|
|
database, err := readDatabase(filepath)
|
|
|
|
if err != nil {
|
|
|
|
// Panic since the file does not exist when Authelia is starting.
|
2019-12-27 17:09:57 +00:00
|
|
|
panic(err.Error())
|
2019-04-24 21:52:08 +00:00
|
|
|
}
|
2019-12-27 17:09:57 +00:00
|
|
|
|
|
|
|
// Early check whether hashed passwords are correct for all users
|
|
|
|
err = checkPasswordHashes(database)
|
|
|
|
if err != nil {
|
|
|
|
panic(err.Error())
|
|
|
|
}
|
|
|
|
|
2019-04-24 21:52:08 +00:00
|
|
|
return &FileUserProvider{
|
|
|
|
path: &filepath,
|
|
|
|
database: database,
|
|
|
|
lock: &sync.Mutex{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-27 17:09:57 +00:00
|
|
|
func checkPasswordHashes(database *DatabaseModel) error {
|
|
|
|
for u, v := range database.Users {
|
|
|
|
_, err := ParseHash(v.HashedPassword)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Unable to parse hash of user %s: %s", u, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-04-24 21:52:08 +00:00
|
|
|
func readDatabase(path string) (*DatabaseModel, error) {
|
|
|
|
content, err := ioutil.ReadFile(path)
|
|
|
|
if err != nil {
|
2019-12-27 17:09:57 +00:00
|
|
|
return nil, fmt.Errorf("Unable to read database from file %s: %s", path, err)
|
2019-04-24 21:52:08 +00:00
|
|
|
}
|
|
|
|
db := DatabaseModel{}
|
|
|
|
err = yaml.Unmarshal(content, &db)
|
|
|
|
if err != nil {
|
2019-12-27 17:09:57 +00:00
|
|
|
return nil, fmt.Errorf("Unable to parse database: %s", err)
|
2019-04-24 21:52:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ok, err := govalidator.ValidateStruct(db)
|
|
|
|
if err != nil {
|
2019-12-27 17:09:57 +00:00
|
|
|
return nil, fmt.Errorf("Invalid schema of database: %s", err)
|
2019-04-24 21:52:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if !ok {
|
2019-12-27 17:09:57 +00:00
|
|
|
return nil, fmt.Errorf("The database format is invalid: %s", err)
|
2019-04-24 21:52:08 +00:00
|
|
|
}
|
|
|
|
return &db, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CheckUserPassword checks if provided password matches for the given user.
|
|
|
|
func (p *FileUserProvider) CheckUserPassword(username string, password string) (bool, error) {
|
|
|
|
if details, ok := p.database.Users[username]; ok {
|
2019-12-27 16:55:00 +00:00
|
|
|
hashedPassword := strings.ReplaceAll(details.HashedPassword, "{CRYPT}", "")
|
2019-04-24 21:52:08 +00:00
|
|
|
ok, err := CheckPassword(password, hashedPassword)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return ok, nil
|
|
|
|
}
|
|
|
|
return false, fmt.Errorf("User '%s' does not exist in database", username)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDetails retrieve the groups a user belongs to.
|
|
|
|
func (p *FileUserProvider) GetDetails(username string) (*UserDetails, error) {
|
|
|
|
if details, ok := p.database.Users[username]; ok {
|
|
|
|
return &UserDetails{
|
|
|
|
Groups: details.Groups,
|
|
|
|
Emails: []string{details.Email},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("User '%s' does not exist in database", username)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdatePassword update the password of the given user.
|
|
|
|
func (p *FileUserProvider) UpdatePassword(username string, newPassword string) error {
|
|
|
|
details, ok := p.database.Users[username]
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("User '%s' does not exist in database", username)
|
|
|
|
}
|
|
|
|
|
2019-11-01 18:31:51 +00:00
|
|
|
hash := HashPassword(newPassword, "")
|
2019-04-24 21:52:08 +00:00
|
|
|
details.HashedPassword = fmt.Sprintf("{CRYPT}%s", hash)
|
|
|
|
|
|
|
|
p.lock.Lock()
|
|
|
|
p.database.Users[username] = details
|
|
|
|
|
|
|
|
b, err := yaml.Marshal(p.database)
|
|
|
|
if err != nil {
|
|
|
|
p.lock.Unlock()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = ioutil.WriteFile(*p.path, b, 0644)
|
|
|
|
p.lock.Unlock()
|
|
|
|
return err
|
|
|
|
}
|