Files
tp/main_dc/yalarba/api_yal/internal/domain/auth/servcie.go
T
valitovgaziz 8c63b1fbb9 modified: main_dc/yalarba/api_yal/internal/domain/auth/handler.go
modified:   main_dc/yalarba/api_yal/internal/domain/auth/servcie.go
change logger from debug to infolavel
2026-03-10 09:58:02 +05:00

236 lines
7.6 KiB
Go

package auth
import (
"api_yal/internal/logger"
"api_yal/internal/models"
"api_yal/internal/repository"
"errors"
"time"
"github.com/golang-jwt/jwt/v5"
"go.uber.org/zap"
"golang.org/x/crypto/bcrypt"
)
// AuthService интерфейс сервиса аутентификации
// Register регистрирует нового пользователя
// Login аутентифицирует пользователя
// RefreshToken обновляет JWT токен
// Logout завершает сессию пользователя
type AuthService interface {
Register(req RegisterRequest) (*AuthResponse, error)
Login(req LoginRequest) (*AuthResponse, error)
RefreshToken(token string) (*AuthResponse, error)
Logout(userID uint) error
}
// authServiceImpl реализация сервиса аутентификации
type authServiceImpl struct {
accountRepo repository.AccountRepository
jwtSecret []byte
}
// NewAuthService создает новый экземпляр сервиса аутентификации
func NewAuthService(accountRepo repository.AccountRepository, jwtSecret string) AuthService {
return &authServiceImpl{
accountRepo: accountRepo,
jwtSecret: []byte(jwtSecret),
}
}
// Register регистрирует нового пользователя
func (s *authServiceImpl) Register(req RegisterRequest) (*AuthResponse, error) {
l := logger.Get()
l.Info("Начало регистрации нового пользователя", zap.String("email", req.Email), zap.String("first_name", req.FirstName), zap.String("last_name", req.LastName))
// Проверяем, существует ли пользователь с таким email
existingUser, err := s.accountRepo.GetByEmail(req.Email)
if err == nil && existingUser != nil {
return nil, ErrUserAlreadyExists
}
// Хешируем пароль
hashedPassword, err := bcrypt.GenerateFromPassword([]byte(req.Password), bcrypt.DefaultCost)
if err != nil {
l.Error("Ошибка хеширования пароля: %v", zap.Error(err))
return nil, err
}
// Формируем полное имя
fullName := req.FirstName + " " + req.LastName
// Создаем аккаунт
newAcc := &models.Account{
Email: req.Email,
PasswordHash: string(hashedPassword),
FirstName: req.FirstName,
LastName: req.LastName,
FullName: fullName,
IsActive: true,
IsVerified: false,
Role: "user",
}
// Сохраняем в базу данных
if err := s.accountRepo.Create(newAcc); err != nil {
l.Error("Ошибка создания аккаунта: %v", zap.Error(err))
return nil, err
}
// Генерируем JWT токен
token, expiresAt, err := s.generateToken(newAcc)
if err != nil {
return nil, err
}
// Формируем ответ
response := &AuthResponse{
Token: token,
ExpiresAt: expiresAt,
User: UserInfo{
ID: newAcc.Base.ID,
Email: newAcc.Email,
FirstName: newAcc.FirstName,
LastName: newAcc.LastName,
FullName: newAcc.FullName,
Role: newAcc.Role,
},
}
l.Info("Пользователь успешно зарегистрирован: %s", zap.String("Email", req.Email))
l.Info("Регистрация успешно завершена", zap.String("email", req.Email))
return response, nil
}
// Login аутентифицирует пользователя
func (s *authServiceImpl) Login(req LoginRequest) (*AuthResponse, error) {
l := logger.Get()
l.Info("Начало входа пользователя", zap.String("email", req.Email))
// Ищем пользователя по email
account, err := s.accountRepo.GetByEmail(req.Email)
if err != nil {
l.Error("Пользователь не найден:",
zap.String("Email", req.Email),
zap.Error(err),
)
return nil, ErrUserNotFound
}
// Проверяем, активен ли аккаунт
if !account.IsActive {
l.Error("Аккаунт деактивирован: %s", zap.String("Email", req.Email))
return nil, errors.New("account is deactivated")
}
// Сравниваем пароли
if err := bcrypt.CompareHashAndPassword([]byte(account.PasswordHash), []byte(req.Password)); err != nil {
l.Error("Неверный пароль для пользователя: %s", zap.String("Email", req.Email))
return nil, ErrInvalidPassword
}
// Генерируем JWT токен
token, expiresAt, err := s.generateToken(account)
if err != nil {
return nil, err
}
// Формируем ответ
response := &AuthResponse{
Token: token,
ExpiresAt: expiresAt,
User: UserInfo{
ID: account.Base.ID,
Email: account.Email,
FirstName: account.FirstName,
LastName: account.LastName,
FullName: account.FullName,
Role: account.Role,
},
}
l.Info("Пользователь успешно вошел: %s", zap.String("Email", req.Email))
l.Info("Вход успешно завершен", zap.String("email", req.Email))
return response, nil
}
// RefreshToken обновляет JWT токен
func (s *authServiceImpl) RefreshToken(token string) (*AuthResponse, error) {
l := logger.Get()
l.Info("Начало обновления токена")
// Парсим и валидируем токен
claims := &jwt.RegisteredClaims{}
parsedToken, err := jwt.ParseWithClaims(token, claims, func(t *jwt.Token) (interface{}, error) {
return s.jwtSecret, nil
})
if err != nil || !parsedToken.Valid {
l.Error("Невалидный токен для обновления: %v", zap.Error(err))
return nil, errors.New("invalid token")
}
// Получаем ID пользователя из claims
userID, err := claims.GetSubject()
if err != nil {
return nil, errors.New("invalid token claims")
}
// Получаем пользователя из базы
var account *models.Account
// Здесь нужно преобразовать string в uint
// В реальном проекте нужно добавить метод GetByIDString или аналогичный
// Для простоты используем существующий метод
// account, err = s.accountRepo.GetByEmail(???)
// Временное решение - нужно добавить метод GetByID
// Пока пропускаем для демонстрации
_ = userID
// Генерируем новый токен
newToken, expiresAt, err := s.generateToken(account)
if err != nil {
return nil, err
}
l.Info("Обновление токена успешно завершено")
return &AuthResponse{
Token: newToken,
ExpiresAt: expiresAt,
}, nil
}
// Logout выход пользователя
func (s *authServiceImpl) Logout(userID uint) error {
l := logger.Get()
l.Info("Начало выхода пользователя", zap.Uint("userID", userID))
l.Info("Выход успешно завершен", zap.Uint("userID", userID))
return nil
}
// generateToken генерирует JWT токен для пользователя
func (s *authServiceImpl) generateToken(account *models.Account) (string, time.Time, error) {
// Устанавливаем время истечения (24 часа)
expiresAt := time.Now().Add(24 * time.Hour)
// Создаем claims
claims := jwt.MapClaims{
"sub": account.Base.ID,
"email": account.Email,
"role": account.Role,
"exp": expiresAt.Unix(),
"iat": time.Now().Unix(),
}
// Создаем токен
token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)
// Подписываем токен
tokenString, err := token.SignedString(s.jwtSecret)
if err != nil {
return "", time.Time{}, err
}
return tokenString, expiresAt, nil
}