88 lines
2.0 KiB
Go
88 lines
2.0 KiB
Go
package store
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/jackc/pgx/v5"
|
|
"github.com/jackc/pgx/v5/pgxpool"
|
|
"vpn-panel/internal/models"
|
|
)
|
|
|
|
var ErrAdminExists = errors.New("администратор уже зарегистрирован")
|
|
|
|
type UserStore struct {
|
|
pool *pgxpool.Pool
|
|
}
|
|
|
|
func NewUserStore(pool *pgxpool.Pool) *UserStore {
|
|
return &UserStore{pool: pool}
|
|
}
|
|
|
|
func (s *UserStore) HasAdmin(ctx context.Context) (bool, error) {
|
|
var n int
|
|
err := s.pool.QueryRow(ctx,
|
|
`SELECT COUNT(*) FROM users WHERE role = 'admin'`,
|
|
).Scan(&n)
|
|
return n > 0, err
|
|
}
|
|
|
|
func (s *UserStore) CreateAdmin(ctx context.Context, email, passwordHash string) (*models.User, error) {
|
|
has, err := s.HasAdmin(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if has {
|
|
return nil, ErrAdminExists
|
|
}
|
|
|
|
var u models.User
|
|
err = s.pool.QueryRow(ctx, `
|
|
INSERT INTO users (email, password_hash, role)
|
|
VALUES ($1, $2, 'admin')
|
|
RETURNING id, email, password_hash, role, created_at
|
|
`, email, passwordHash).Scan(
|
|
&u.ID, &u.Email, &u.PasswordHash, &u.Role, &u.CreatedAt,
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &u, nil
|
|
}
|
|
|
|
func (s *UserStore) GetByEmail(ctx context.Context, email string) (*models.User, error) {
|
|
var u models.User
|
|
err := s.pool.QueryRow(ctx, `
|
|
SELECT id, email, password_hash, role, created_at
|
|
FROM users WHERE email = $1
|
|
`, email).Scan(&u.ID, &u.Email, &u.PasswordHash, &u.Role, &u.CreatedAt)
|
|
if errors.Is(err, pgx.ErrNoRows) {
|
|
return nil, nil
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &u, nil
|
|
}
|
|
|
|
func (s *UserStore) CountUsers(ctx context.Context) (int, error) {
|
|
var n int
|
|
err := s.pool.QueryRow(ctx, `SELECT COUNT(*) FROM users`).Scan(&n)
|
|
return n, err
|
|
}
|
|
|
|
func (s *UserStore) GetAdminID(ctx context.Context) (uuid.UUID, bool, error) {
|
|
var id uuid.UUID
|
|
err := s.pool.QueryRow(ctx,
|
|
`SELECT id FROM users WHERE role = 'admin' LIMIT 1`,
|
|
).Scan(&id)
|
|
if errors.Is(err, pgx.ErrNoRows) {
|
|
return uuid.Nil, false, nil
|
|
}
|
|
if err != nil {
|
|
return uuid.Nil, false, err
|
|
}
|
|
return id, true, nil
|
|
}
|