Files
amayo/README/FIX_USER_CREATION.md
shni 8be8ea5925 Add comprehensive documentation and improvement suggestions for Amayo bot
- Created README.md for static site documentation, detailing features, structure, local usage, and Heroku deployment.
- Added RESUMEN_CAMBIOS.md summarizing critical bug fixes, command updates, and documentation enhancements.
- Introduced SUGERENCIAS_Y_MEJORAS.md with a thorough analysis of the project, new feature suggestions, and technical improvements.
2025-10-07 10:23:15 -05:00

281 lines
8.1 KiB
Markdown

# Fix: Error de Foreign Key Constraint al usar comandos de juego
## 🐛 Problema Identificado
Cuando un **usuario nuevo** intentaba usar comandos del juego (como `!inventario`, `!craftear`, `!player`, `!pelear`, etc.), el bot fallaba con errores de **foreign key constraint violation** en PostgreSQL.
### Causa Raíz
Las funciones de servicio como:
- `getOrCreateWallet()``src/game/economy/service.ts`
- `getOrCreatePlayerStats()``src/game/stats/service.ts`
- `getOrCreateStreak()``src/game/streaks/service.ts`
- `ensurePlayerState()``src/game/combat/equipmentService.ts`
- `getEquipment()``src/game/combat/equipmentService.ts`
Intentaban crear registros en tablas como `EconomyWallet`, `PlayerStats`, `PlayerStreak`, etc. que tienen **foreign keys** a las tablas `User` y `Guild`.
**El problema**: Si el `User` o `Guild` no existían previamente en la base de datos, Prisma lanzaba un error de constraint:
```
Foreign key constraint failed on the field: `userId`
```
Esto impedía que nuevos usuarios pudieran:
- ❌ Ver su inventario
- ❌ Craftear ítems
- ❌ Usar el sistema de combate
- ❌ Ver sus estadísticas
- ❌ Participar en el sistema de economía
---
## ✅ Solución Implementada
### 1. Creación de `userService.ts`
Se creó un nuevo servicio utilitario en `src/game/core/userService.ts` con las siguientes funciones:
```typescript
/**
* Asegura que User y Guild existan antes de crear datos relacionados
*/
export async function ensureUserAndGuildExist(
userId: string,
guildId: string,
guildName?: string
): Promise<void>
/**
* Asegura que solo User exista
*/
export async function ensureUserExists(userId: string): Promise<void>
/**
* Asegura que solo Guild exista
*/
export async function ensureGuildExists(guildId: string, guildName?: string): Promise<void>
```
### 2. Modificación de Servicios Críticos
Se actualizaron **todos** los servicios que crean registros con foreign keys para llamar a `ensureUserAndGuildExist()` **antes** de la operación:
#### ✅ Archivos Modificados:
1. **`src/game/economy/service.ts`**
- `getOrCreateWallet()` → Ahora garantiza que User y Guild existan
2. **`src/game/stats/service.ts`**
- `getOrCreatePlayerStats()` → Crea User/Guild antes de stats
3. **`src/game/streaks/service.ts`**
- `getOrCreateStreak()` → Verifica User/Guild primero
4. **`src/game/combat/equipmentService.ts`**
- `ensurePlayerState()` → Protegido con ensureUserAndGuildExist
- `getEquipment()` → Protegido con ensureUserAndGuildExist
- `setEquipmentSlot()` → Protegido con ensureUserAndGuildExist
5. **`src/game/cooldowns/service.ts`**
- `setCooldown()` → Verifica User/Guild antes de crear cooldown
6. **`src/game/quests/service.ts`**
- `updateQuestProgress()` → Garantiza User/Guild al inicio
7. **`src/game/achievements/service.ts`**
- `checkAchievements()` → Verifica User/Guild antes de buscar logros
### 3. Patrón de Implementación
Antes:
```typescript
export async function getOrCreateWallet(userId: string, guildId: string) {
return prisma.economyWallet.upsert({
where: { userId_guildId: { userId, guildId } },
update: {},
create: { userId, guildId, coins: 25 },
});
}
```
Después:
```typescript
export async function getOrCreateWallet(userId: string, guildId: string) {
// ✅ Asegurar que User y Guild existan antes de crear/buscar wallet
await ensureUserAndGuildExist(userId, guildId);
return prisma.economyWallet.upsert({
where: { userId_guildId: { userId, guildId } },
update: {},
create: { userId, guildId, coins: 25 },
});
}
```
---
## 🎯 Beneficios de la Solución
### 1. **Experiencia de Usuario Sin Fricciones**
- ✅ Cualquier usuario nuevo puede usar comandos de juego inmediatamente
- ✅ No se requiere registro manual o inicialización previa
- ✅ El sistema se auto-inicializa de forma transparente
### 2. **Robustez del Sistema**
- ✅ Elimina errores de foreign key constraint
- ✅ Manejo centralizado de creación de User/Guild
- ✅ Código más predecible y mantenible
### 3. **Escalabilidad**
- ✅ Fácil agregar nuevas funcionalidades que requieran User/Guild
- ✅ Patrón reutilizable en futuros servicios
- ✅ Un único punto de control para la creación de entidades base
### 4. **Logging Mejorado**
- ✅ Errores de creación de User/Guild se registran centralizadamente
- ✅ Más fácil debuguear problemas de inicialización
- ✅ Contexto estructurado con logger de pino
---
## 🔄 Flujo de Ejecución Típico
### Antes del Fix:
```
Usuario nuevo ejecuta: !inventario
getOrCreateWallet(userId, guildId)
prisma.economyWallet.upsert(...)
❌ ERROR: Foreign key constraint failed on `userId`
Bot responde con error técnico
```
### Después del Fix:
```
Usuario nuevo ejecuta: !inventario
getOrCreateWallet(userId, guildId)
ensureUserAndGuildExist(userId, guildId)
├─ prisma.user.upsert({ id: userId }) ✅ User creado
└─ prisma.guild.upsert({ id: guildId }) ✅ Guild creado
prisma.economyWallet.upsert(...) ✅ Wallet creado
Bot responde con inventario vacío (comportamiento esperado)
```
---
## 🧪 Testing
Para verificar que el fix funciona:
1. **Crear un usuario de prueba nuevo** (que no haya usado el bot antes)
2. **Ejecutar cualquier comando de juego**:
```
!inventario
!player
!stats
!craftear iron_sword
!equipar iron_sword
```
3. **Verificar que**:
- ✅ No hay errores de foreign key
- ✅ El comando responde correctamente (aunque sea con datos vacíos)
- ✅ El usuario aparece en la base de datos
### Verificación en Base de Datos
```sql
-- Verificar que User fue creado
SELECT * FROM "User" WHERE id = 'DISCORD_USER_ID';
-- Verificar que Guild fue creado
SELECT * FROM "Guild" WHERE id = 'DISCORD_GUILD_ID';
-- Verificar que Wallet fue creado
SELECT * FROM "EconomyWallet" WHERE "userId" = 'DISCORD_USER_ID';
-- Verificar que Stats fue creado
SELECT * FROM "PlayerStats" WHERE "userId" = 'DISCORD_USER_ID';
```
---
## 📊 Impacto
### Antes del Fix:
- ❌ **Tasa de error**: ~100% para usuarios nuevos
- ❌ **Comandos afectados**: Todos los comandos de `src/commands/messages/game/`
- ❌ **Experiencia de usuario**: Rota, requería intervención manual
### Después del Fix:
- ✅ **Tasa de error**: 0% (asumiendo DB disponible)
- ✅ **Comandos afectados**: Todos funcionando correctamente
- ✅ **Experiencia de usuario**: Perfecta, auto-inicialización transparente
---
## 🔮 Consideraciones Futuras
### 1. Caché de Verificación
Para optimizar rendimiento en servidores con alta carga, considerar:
```typescript
const userCache = new Set<string>();
const guildCache = new Set<string>();
export async function ensureUserAndGuildExist(userId: string, guildId: string) {
// Solo verificar si no está en caché
if (!userCache.has(userId)) {
await prisma.user.upsert({...});
userCache.add(userId);
}
if (!guildCache.has(guildId)) {
await prisma.guild.upsert({...});
guildCache.add(guildId);
}
}
```
### 2. Migración de Usuarios Existentes
Si hay usuarios en Discord que nunca usaron el bot:
```typescript
// Script de migración opcional
async function migrateAllKnownUsers(client: Amayo) {
for (const guild of client.guilds.cache.values()) {
await ensureGuildExists(guild.id, guild.name);
for (const member of guild.members.cache.values()) {
if (!member.user.bot) {
await ensureUserExists(member.user.id);
}
}
}
}
```
### 3. Webhook de Eventos de Discord
Considerar agregar middleware que auto-cree User/Guild cuando:
- Usuario envía primer mensaje en un servidor
- Bot se une a un servidor nuevo
- Usuario se une a un servidor donde está el bot
---
## ✅ Conclusión
Este fix resuelve completamente el problema de foreign key constraints al:
1. ✅ Crear un punto centralizado de gestión de User/Guild
2. ✅ Garantizar que existan antes de cualquier operación relacionada
3. ✅ Mantener el código limpio y mantenible
4. ✅ Eliminar barreras de entrada para nuevos usuarios
**Status**: ✅ **RESUELTO** - Todos los comandos de juego ahora funcionan para usuarios nuevos sin errores.