216 lines
6.9 KiB
Plaintext
216 lines
6.9 KiB
Plaintext
// AR-ProjectManagement — Prisma Schema
|
|
// Base de datos: PostgreSQL 16
|
|
|
|
generator client {
|
|
provider = "prisma-client-js"
|
|
}
|
|
|
|
datasource db {
|
|
provider = "postgresql"
|
|
url = env("DATABASE_URL")
|
|
}
|
|
|
|
// ============================================================
|
|
// ENUMS
|
|
// ============================================================
|
|
enum Role {
|
|
SUPER_ADMIN
|
|
DIRECTOR
|
|
GERENTE
|
|
JEFE_AREA
|
|
COLABORADOR
|
|
FINANCIERO
|
|
AUDITOR
|
|
}
|
|
|
|
enum ProjectStatus {
|
|
ACTIVO
|
|
PAUSADO
|
|
COMPLETADO
|
|
EN_RIESGO
|
|
CANCELADO
|
|
}
|
|
|
|
enum TaskStatus {
|
|
PENDIENTE
|
|
EN_PROCESO
|
|
BLOQUEADA
|
|
EN_VERIFICACION
|
|
NO_CONFORME
|
|
APROBADO_TECNICO
|
|
ACEPTADO_FORMAL
|
|
COMPLETADA
|
|
}
|
|
|
|
enum Priority {
|
|
CRITICO
|
|
ALTO
|
|
MEDIO
|
|
BAJO
|
|
}
|
|
|
|
// ============================================================
|
|
// MODULO 1: USUARIOS Y AUTENTICACION
|
|
// ============================================================
|
|
model User {
|
|
id String @id @default(uuid())
|
|
email String @unique
|
|
passwordHash String @map("password_hash")
|
|
firstName String @map("first_name")
|
|
lastName String @map("last_name")
|
|
role Role @default(COLABORADOR)
|
|
isActive Boolean @default(true) @map("is_active")
|
|
isTwoFactorEnabled Boolean @default(false) @map("is_two_factor_enabled")
|
|
twoFactorSecret String? @map("two_factor_secret")
|
|
loginAttempts Int @default(0) @map("login_attempts")
|
|
lockedUntil DateTime? @map("locked_until")
|
|
lastLoginAt DateTime? @map("last_login_at")
|
|
lastLoginIp String? @map("last_login_ip")
|
|
avatarUrl String? @map("avatar_url")
|
|
phone String?
|
|
department String?
|
|
jobTitle String? @map("job_title")
|
|
createdAt DateTime @default(now()) @map("created_at")
|
|
updatedAt DateTime @updatedAt @map("updated_at")
|
|
|
|
refreshTokens RefreshToken[]
|
|
auditLogs AuditLog[]
|
|
projectMembers ProjectMember[]
|
|
tasksAssigned Task[] @relation("TaskAssignee")
|
|
tasksCreated Task[] @relation("TaskCreator")
|
|
comments Comment[]
|
|
|
|
@@map("users")
|
|
}
|
|
|
|
model RefreshToken {
|
|
id String @id @default(uuid())
|
|
token String @unique
|
|
userId String @map("user_id")
|
|
expiresAt DateTime @map("expires_at")
|
|
createdAt DateTime @default(now()) @map("created_at")
|
|
revokedAt DateTime? @map("revoked_at")
|
|
ipAddress String? @map("ip_address")
|
|
userAgent String? @map("user_agent")
|
|
|
|
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
|
|
@@map("refresh_tokens")
|
|
}
|
|
|
|
// ============================================================
|
|
// LOG DE AUDITORIA INMUTABLE
|
|
// ============================================================
|
|
model AuditLog {
|
|
id String @id @default(uuid())
|
|
userId String? @map("user_id")
|
|
action String
|
|
entity String
|
|
entityId String? @map("entity_id")
|
|
oldValues Json? @map("old_values")
|
|
newValues Json? @map("new_values")
|
|
ipAddress String? @map("ip_address")
|
|
userAgent String? @map("user_agent")
|
|
createdAt DateTime @default(now()) @map("created_at")
|
|
|
|
user User? @relation(fields: [userId], references: [id], onDelete: SetNull)
|
|
|
|
@@map("audit_logs")
|
|
}
|
|
|
|
// ============================================================
|
|
// MODULO 3: PROYECTOS
|
|
// ============================================================
|
|
model Project {
|
|
id String @id @default(uuid())
|
|
code String @unique
|
|
name String
|
|
description String?
|
|
type String
|
|
category String
|
|
client String?
|
|
startDate DateTime @map("start_date")
|
|
plannedEndDate DateTime @map("planned_end_date")
|
|
actualEndDate DateTime? @map("actual_end_date")
|
|
budget Decimal @db.Decimal(15, 2)
|
|
contingencyPct Decimal @default(10) @map("contingency_pct") @db.Decimal(5, 2)
|
|
currency String @default("USD")
|
|
priority Priority @default(MEDIO)
|
|
status ProjectStatus @default(ACTIVO)
|
|
currentPhase String? @map("current_phase")
|
|
methodology String?
|
|
tags String[]
|
|
ragStatus String @default("VERDE") @map("rag_status")
|
|
progressPct Decimal @default(0) @map("progress_pct") @db.Decimal(5, 2)
|
|
createdAt DateTime @default(now()) @map("created_at")
|
|
updatedAt DateTime @updatedAt @map("updated_at")
|
|
|
|
members ProjectMember[]
|
|
tasks Task[]
|
|
|
|
@@map("projects")
|
|
}
|
|
|
|
model ProjectMember {
|
|
id String @id @default(uuid())
|
|
projectId String @map("project_id")
|
|
userId String @map("user_id")
|
|
role String
|
|
joinedAt DateTime @default(now()) @map("joined_at")
|
|
|
|
project Project @relation(fields: [projectId], references: [id], onDelete: Cascade)
|
|
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
|
|
@@unique([projectId, userId])
|
|
@@map("project_members")
|
|
}
|
|
|
|
// ============================================================
|
|
// MODULO 4: TAREAS
|
|
// ============================================================
|
|
model Task {
|
|
id String @id @default(uuid())
|
|
projectId String @map("project_id")
|
|
parentId String? @map("parent_id")
|
|
wbsId String? @map("wbs_id")
|
|
title String
|
|
description String?
|
|
assigneeId String? @map("assignee_id")
|
|
createdById String @map("created_by_id")
|
|
startDate DateTime? @map("start_date")
|
|
dueDate DateTime? @map("due_date")
|
|
estimatedHours Decimal? @map("estimated_hours") @db.Decimal(8, 2)
|
|
actualHours Decimal? @map("actual_hours") @db.Decimal(8, 2)
|
|
progressPct Decimal @default(0) @map("progress_pct") @db.Decimal(5, 2)
|
|
status TaskStatus @default(PENDIENTE)
|
|
priority Priority @default(MEDIO)
|
|
estimatedCost Decimal? @map("estimated_cost") @db.Decimal(15, 2)
|
|
actualCost Decimal? @map("actual_cost") @db.Decimal(15, 2)
|
|
tags String[]
|
|
createdAt DateTime @default(now()) @map("created_at")
|
|
updatedAt DateTime @updatedAt @map("updated_at")
|
|
|
|
project Project @relation(fields: [projectId], references: [id], onDelete: Cascade)
|
|
assignee User? @relation("TaskAssignee", fields: [assigneeId], references: [id])
|
|
creator User @relation("TaskCreator", fields: [createdById], references: [id])
|
|
parent Task? @relation("TaskChildren", fields: [parentId], references: [id])
|
|
children Task[] @relation("TaskChildren")
|
|
comments Comment[]
|
|
|
|
@@map("tasks")
|
|
}
|
|
|
|
model Comment {
|
|
id String @id @default(uuid())
|
|
taskId String @map("task_id")
|
|
userId String @map("user_id")
|
|
content String
|
|
createdAt DateTime @default(now()) @map("created_at")
|
|
updatedAt DateTime @updatedAt @map("updated_at")
|
|
|
|
task Task @relation(fields: [taskId], references: [id], onDelete: Cascade)
|
|
user User @relation(fields: [userId], references: [id])
|
|
|
|
@@map("comments")
|
|
}
|