2020-06-22 15:16:26 +02:00
|
|
|
package entity
|
|
|
|
|
|
|
|
import (
|
|
|
|
"time"
|
|
|
|
|
2020-06-27 14:15:25 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/txt"
|
2020-06-22 15:16:26 +02:00
|
|
|
"golang.org/x/crypto/bcrypt"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Password represents a password hash.
|
|
|
|
type Password struct {
|
|
|
|
UID string `gorm:"type:varbinary(255);primary_key;" json:"UID"`
|
|
|
|
Hash string `deepcopier:"skip" gorm:"type:varbinary(255);" json:"Hash"`
|
|
|
|
CreatedAt time.Time `deepcopier:"skip" json:"CreatedAt"`
|
|
|
|
UpdatedAt time.Time `deepcopier:"skip" json:"UpdatedAt"`
|
|
|
|
}
|
|
|
|
|
2020-06-25 01:20:58 +02:00
|
|
|
// NewPassword creates a new password instance.
|
2020-06-22 15:16:26 +02:00
|
|
|
func NewPassword(uid, password string) Password {
|
|
|
|
if uid == "" {
|
2020-06-27 14:15:25 +02:00
|
|
|
panic("auth: can't set password without uid")
|
2020-06-22 15:16:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
m := Password{UID: uid}
|
|
|
|
|
|
|
|
if password != "" {
|
|
|
|
if err := m.SetPassword(password); err != nil {
|
2020-06-27 14:15:25 +02:00
|
|
|
log.Errorf("auth: failed setting password for %s", uid)
|
2020-06-22 15:16:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
2020-06-25 01:20:58 +02:00
|
|
|
// SetPassword sets a new password stored as hash.
|
2020-06-22 15:16:26 +02:00
|
|
|
func (m *Password) SetPassword(password string) error {
|
|
|
|
if bytes, err := bcrypt.GenerateFromPassword([]byte(password), 14); err != nil {
|
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
m.Hash = string(bytes)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-25 01:20:58 +02:00
|
|
|
// InvalidPassword returns true if the given password does not match the hash.
|
2020-06-22 15:16:26 +02:00
|
|
|
func (m *Password) InvalidPassword(password string) bool {
|
|
|
|
if m.Hash == "" && password == "" {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
err := bcrypt.CompareHashAndPassword([]byte(m.Hash), []byte(password))
|
|
|
|
return err != nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create inserts a new row to the database.
|
|
|
|
func (m *Password) Create() error {
|
|
|
|
return Db().Create(m).Error
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save inserts a new row to the database or updates a row if the primary key already exists.
|
|
|
|
func (m *Password) Save() error {
|
|
|
|
return Db().Save(m).Error
|
|
|
|
}
|
|
|
|
|
|
|
|
// FindPassword returns an entity pointer if exists.
|
|
|
|
func FindPassword(uid string) *Password {
|
|
|
|
result := Password{}
|
|
|
|
|
|
|
|
if err := Db().Where("uid = ?", uid).First(&result).Error; err == nil {
|
|
|
|
return &result
|
|
|
|
} else {
|
2020-06-27 14:15:25 +02:00
|
|
|
log.Errorf("auth: no password for %s", txt.Quote(uid))
|
2020-06-22 15:16:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// String returns the password hash.
|
|
|
|
func (m *Password) String() string {
|
|
|
|
return m.Hash
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unknown returns true if the password is an empty string.
|
|
|
|
func (m *Password) Unknown() bool {
|
|
|
|
return m.Hash == ""
|
|
|
|
}
|