package postgres import ( "context" "database/sql" "encoding/json" "time" "github.com/google/uuid" "github.com/rede5/gohorsejobs/backend/internal/core/domain/entity" ) type UserRepository struct { db *sql.DB } func NewUserRepository(db *sql.DB) *UserRepository { return &UserRepository{db: db} } func (r *UserRepository) Save(ctx context.Context, user *entity.User) (*entity.User, error) { if user.ID == "" { user.ID = uuid.New().String() } tx, err := r.db.BeginTx(ctx, nil) if err != nil { return nil, err } defer tx.Rollback() // 1. Insert User query := ` INSERT INTO core_users (id, tenant_id, name, email, password_hash, status, created_at, updated_at) VALUES ($1, $2, $3, $4, $5, $6, $7, $8) ` _, err = tx.ExecContext(ctx, query, user.ID, user.TenantID, user.Name, user.Email, user.PasswordHash, user.Status, user.CreatedAt, user.UpdatedAt, ) if err != nil { return nil, err } // 2. Insert Roles if len(user.Roles) > 0 { roleQuery := `INSERT INTO core_user_roles (user_id, role) VALUES ($1, $2)` for _, role := range user.Roles { _, err := tx.ExecContext(ctx, roleQuery, user.ID, role.Name) if err != nil { return nil, err } } } if err := tx.Commit(); err != nil { return nil, err } return user, nil } func (r *UserRepository) FindByEmail(ctx context.Context, email string) (*entity.User, error) { query := ` SELECT id, tenant_id, name, email, password_hash, status, created_at, updated_at, bio, profile_picture_url, skills, experience, education FROM core_users WHERE email = $1 ` row := r.db.QueryRowContext(ctx, query, email) u := &entity.User{} var skills, experience, education []byte // temp for Scanning err := row.Scan( &u.ID, &u.TenantID, &u.Name, &u.Email, &u.PasswordHash, &u.Status, &u.CreatedAt, &u.UpdatedAt, &u.Bio, &u.ProfilePictureURL, &skills, &experience, &education, ) if err != nil { if err == sql.ErrNoRows { return nil, nil // Return nil if not found } return nil, err } // Unmarshal JSONB fields if len(skills) > 0 { _ = json.Unmarshal(skills, &u.Skills) } if len(experience) > 0 { _ = json.Unmarshal(experience, &u.Experience) } if len(education) > 0 { _ = json.Unmarshal(education, &u.Education) } u.Roles, _ = r.getRoles(ctx, u.ID) return u, nil } func (r *UserRepository) FindByID(ctx context.Context, id string) (*entity.User, error) { query := ` SELECT id, tenant_id, name, email, password_hash, status, created_at, updated_at, bio, profile_picture_url, skills, experience, education FROM core_users WHERE id = $1 ` row := r.db.QueryRowContext(ctx, query, id) u := &entity.User{} var skills, experience, education []byte // temp for Scanning err := row.Scan( &u.ID, &u.TenantID, &u.Name, &u.Email, &u.PasswordHash, &u.Status, &u.CreatedAt, &u.UpdatedAt, &u.Bio, &u.ProfilePictureURL, &skills, &experience, &education, ) if err != nil { return nil, err } // Unmarshal JSONB fields if len(skills) > 0 { _ = json.Unmarshal(skills, &u.Skills) } if len(experience) > 0 { _ = json.Unmarshal(experience, &u.Experience) } if len(education) > 0 { _ = json.Unmarshal(education, &u.Education) } u.Roles, _ = r.getRoles(ctx, u.ID) return u, nil } func (r *UserRepository) FindAllByTenant(ctx context.Context, tenantID string) ([]*entity.User, error) { query := `SELECT id, tenant_id, name, email, password_hash, status, created_at, updated_at FROM core_users WHERE tenant_id = $1` rows, err := r.db.QueryContext(ctx, query, tenantID) if err != nil { return nil, err } defer rows.Close() var users []*entity.User for rows.Next() { u := &entity.User{} if err := rows.Scan(&u.ID, &u.TenantID, &u.Name, &u.Email, &u.PasswordHash, &u.Status, &u.CreatedAt, &u.UpdatedAt); err != nil { return nil, err } u.Roles, _ = r.getRoles(ctx, u.ID) users = append(users, u) } return users, nil } func (r *UserRepository) Update(ctx context.Context, user *entity.User) (*entity.User, error) { user.UpdatedAt = time.Now() skillsJSON, _ := json.Marshal(user.Skills) experienceJSON, _ := json.Marshal(user.Experience) educationJSON, _ := json.Marshal(user.Education) query := ` UPDATE core_users SET name=$1, email=$2, status=$3, updated_at=$4, bio=$5, profile_picture_url=$6, skills=$7, experience=$8, education=$9 WHERE id=$10 ` _, err := r.db.ExecContext(ctx, query, user.Name, user.Email, user.Status, user.UpdatedAt, user.Bio, user.ProfilePictureURL, skillsJSON, experienceJSON, educationJSON, user.ID, ) return user, err } func (r *UserRepository) Delete(ctx context.Context, id string) error { _, err := r.db.ExecContext(ctx, `DELETE FROM core_users WHERE id=$1`, id) return err } func (r *UserRepository) getRoles(ctx context.Context, userID string) ([]entity.Role, error) { rows, err := r.db.QueryContext(ctx, `SELECT role FROM core_user_roles WHERE user_id = $1`, userID) if err != nil { return nil, err } defer rows.Close() var roles []entity.Role for rows.Next() { var roleName string rows.Scan(&roleName) roles = append(roles, entity.Role{Name: roleName}) } return roles, nil }