2024-02-07 19:07:29 +00:00
|
|
|
package models
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
2024-02-07 23:54:12 +00:00
|
|
|
"errors"
|
|
|
|
"strings"
|
2024-02-07 23:15:54 +00:00
|
|
|
"time"
|
2024-02-07 23:54:12 +00:00
|
|
|
|
|
|
|
"github.com/go-sql-driver/mysql"
|
|
|
|
"golang.org/x/crypto/bcrypt"
|
2024-02-07 19:07:29 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type User struct {
|
2024-02-07 23:15:54 +00:00
|
|
|
ID int
|
|
|
|
Username string
|
|
|
|
Email string
|
|
|
|
HashedPassword []byte
|
|
|
|
Created time.Time
|
2024-02-07 19:07:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type UserModel struct {
|
|
|
|
DB *sql.DB
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert
|
2024-02-07 23:54:12 +00:00
|
|
|
func (m *UserModel) Insert(username, email, password string) error {
|
|
|
|
hashedPassword, err := bcrypt.GenerateFromPassword([]byte(password), 15)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
stmt := `INSERT INTO users (username, email, hashed_password, created)
|
|
|
|
VALUES(?, ?, ?, UTC_TIMESTAMP())`
|
|
|
|
|
|
|
|
_, err = m.DB.Exec(stmt, username, email, string(hashedPassword))
|
|
|
|
if err != nil {
|
|
|
|
var mySQLError *mysql.MySQLError
|
|
|
|
if errors.As(err, &mySQLError) {
|
|
|
|
if mySQLError.Number == 1062 && strings.Contains(mySQLError.Message, "users_uc_email") {
|
|
|
|
return ErrDuplicateEmail
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2024-02-07 19:07:29 +00:00
|
|
|
}
|
|
|
|
|
2024-02-07 23:15:54 +00:00
|
|
|
// Authenticate
|
2024-02-08 00:27:03 +00:00
|
|
|
func (m *UserModel) Authenticate(email, password string) (int, error) {
|
|
|
|
var id int
|
|
|
|
var hashedPassword []byte
|
|
|
|
|
|
|
|
stmt := "SELECT id, hashed_password FROM users WHERE email = ?"
|
|
|
|
|
|
|
|
err := m.DB.QueryRow(stmt, email).Scan(&id, &hashedPassword)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, sql.ErrNoRows) {
|
|
|
|
return 0, ErrInvalidCredentials
|
|
|
|
} else {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = bcrypt.CompareHashAndPassword(hashedPassword, []byte(password))
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, bcrypt.ErrMismatchedHashAndPassword) {
|
|
|
|
return 0, ErrInvalidCredentials
|
|
|
|
} else {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return id, nil
|
2024-02-07 23:15:54 +00:00
|
|
|
}
|
2024-02-07 19:07:29 +00:00
|
|
|
|
2024-02-07 23:15:54 +00:00
|
|
|
// Exists
|
|
|
|
func (m *UserModel) Exists(id int) (bool, error) {
|
2024-02-08 17:26:25 +00:00
|
|
|
var exists bool
|
|
|
|
|
|
|
|
stmt := "SELECT EXISTS(SELECT true FROM users WHERE id = ?)"
|
|
|
|
|
|
|
|
err := m.DB.QueryRow(stmt, id).Scan(&exists)
|
|
|
|
return exists, err
|
2024-02-07 19:07:29 +00:00
|
|
|
}
|