Files
econ/backend/internal/repository/progreso.go
Renato d31575a143 Initial commit: Plataforma de Economía
Features:
- React 18 + TypeScript frontend with Vite
- Go + Gin backend API
- PostgreSQL database
- JWT authentication with refresh tokens
- User management (admin panel)
- Docker containerization
- Progress tracking system
- 4 economic modules structure

Fixed:
- Login with username or email
- User creation without required email
- Database nullable timestamps
- API response field naming
2026-02-12 01:30:57 +01:00

142 lines
4.5 KiB
Go

package repository
import (
"context"
"time"
"github.com/google/uuid"
"github.com/jackc/pgx/v5/pgxpool"
"github.com/ren/econ/backend/internal/models"
)
type ProgresoRepository struct {
db *pgxpool.Pool
}
func NewProgresoRepository(db *pgxpool.Pool) *ProgresoRepository {
return &ProgresoRepository{db: db}
}
func (r *ProgresoRepository) GetByUsuario(ctx context.Context, usuarioID uuid.UUID) ([]models.Progreso, error) {
query := `
SELECT id, usuario_id, modulo_numero, ejercicio_id, completado, puntuacion, intentos, ultima_vez, respuesta_json
FROM progreso_usuario WHERE usuario_id = $1
ORDER BY ultima_vez DESC
`
rows, err := r.db.Query(ctx, query, usuarioID)
if err != nil {
return nil, err
}
defer rows.Close()
var progresos []models.Progreso
for rows.Next() {
var p models.Progreso
err := rows.Scan(
&p.ID, &p.UsuarioID, &p.ModuloNumero, &p.EjercicioID,
&p.Completado, &p.Puntuacion, &p.Intentos, &p.UltimaVez, &p.RespuestaJSON)
if err != nil {
return nil, err
}
progresos = append(progresos, p)
}
return progresos, nil
}
func (r *ProgresoRepository) GetByModulo(ctx context.Context, usuarioID uuid.UUID, moduloNumero int) ([]models.Progreso, error) {
query := `
SELECT id, usuario_id, modulo_numero, ejercicio_id, completado, puntuacion, intentos, ultima_vez, respuesta_json
FROM progreso_usuario WHERE usuario_id = $1 AND modulo_numero = $2
ORDER BY ejercicio_id
`
rows, err := r.db.Query(ctx, query, usuarioID, moduloNumero)
if err != nil {
return nil, err
}
defer rows.Close()
var progresos []models.Progreso
for rows.Next() {
var p models.Progreso
err := rows.Scan(
&p.ID, &p.UsuarioID, &p.ModuloNumero, &p.EjercicioID,
&p.Completado, &p.Puntuacion, &p.Intentos, &p.UltimaVez, &p.RespuestaJSON)
if err != nil {
return nil, err
}
progresos = append(progresos, p)
}
return progresos, nil
}
func (r *ProgresoRepository) GetByEjercicio(ctx context.Context, usuarioID uuid.UUID, ejercicioID int) (*models.Progreso, error) {
query := `
SELECT id, usuario_id, modulo_numero, ejercicio_id, completado, puntuacion, intentos, ultima_vez, respuesta_json
FROM progreso_usuario WHERE usuario_id = $1 AND ejercicio_id = $2
`
var p models.Progreso
err := r.db.QueryRow(ctx, query, usuarioID, ejercicioID).Scan(
&p.ID, &p.UsuarioID, &p.ModuloNumero, &p.EjercicioID,
&p.Completado, &p.Puntuacion, &p.Intentos, &p.UltimaVez, &p.RespuestaJSON)
if err != nil {
return nil, err
}
return &p, nil
}
func (r *ProgresoRepository) Upsert(ctx context.Context, usuarioID uuid.UUID, ejercicioID int, update *models.ProgresoUpdate) error {
query := `
INSERT INTO progreso_usuario (id, usuario_id, modulo_numero, ejercicio_id, completado, puntuacion, intentos, ultima_vez, respuesta_json)
VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9)
ON CONFLICT (usuario_id, modulo_numero, ejercicio_id)
DO UPDATE SET completado = $5, puntuacion = $6, intentos = $7, ultima_vez = $8, respuesta_json = $9
`
moduloNumero, err := r.getModuloByEjercicio(ctx, ejercicioID)
if err != nil {
return err
}
existing, _ := r.GetByEjercicio(ctx, usuarioID, ejercicioID)
var intentos int
if existing != nil {
intentos = existing.Intentos + 1
} else {
intentos = 1
}
_, err = r.db.Exec(ctx, query,
uuid.New(), usuarioID, moduloNumero, ejercicioID,
update.Completado, update.Puntuacion, intentos, time.Now(), update.RespuestaJSON)
return err
}
func (r *ProgresoRepository) getModuloByEjercicio(ctx context.Context, ejercicioID int) (int, error) {
var moduloNumero int
err := r.db.QueryRow(ctx, "SELECT modulo_numero FROM ejercicios WHERE id = $1", ejercicioID).Scan(&moduloNumero)
return moduloNumero, err
}
func (r *ProgresoRepository) GetResumen(ctx context.Context, usuarioID uuid.UUID) (*models.ProgresoResumen, error) {
query := `
SELECT
COUNT(DISTINCT ejercicio_id) as total,
COUNT(CASE WHEN completado THEN 1 END) as completados,
COALESCE(AVG(CASE WHEN completado THEN puntuacion END), 0)::int as promedio,
COUNT(DISTINCT CASE WHEN completado THEN modulo_numero END) as modulos
FROM progreso_usuario WHERE usuario_id = $1
`
var resumen models.ProgresoResumen
err := r.db.QueryRow(ctx, query, usuarioID).Scan(
&resumen.TotalEjercicios, &resumen.EjerciciosCompletados,
&resumen.PromedioPuntuacion, &resumen.ModulosCompletados)
if err != nil {
return nil, err
}
return &resumen, nil
}
func (r *ProgresoRepository) GetByUsuarioID(ctx context.Context, usuarioID uuid.UUID) ([]models.Progreso, error) {
return r.GetByUsuario(ctx, usuarioID)
}