codigo0/generar_indice_app.py

545 lines
27 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
"""
Genera el archivo manual-index.ts desde la estructura real de archivos
"""
import json
from pathlib import Path
BASE_DIR = Path("/home/planetazuzu/protocolo-r-pido")
MANUAL_DIR = BASE_DIR / "manual-tes" / "TES_Manual_Digital"
# Mapeo del índice
INDICE_ESPERADO = {
"1.1.1": {"nombre": "Fundamentos de Emergencias", "bloque": 0, "parte": 1},
"1.2.1": {"nombre": "Constantes Vitales", "bloque": 1, "parte": 1},
"1.2.2": {"nombre": "ABCDE Operativo", "bloque": 1, "parte": 1},
"1.2.3": {"nombre": "Glasgow Operativo", "bloque": 1, "parte": 1},
"1.2.4": {"nombre": "Triage START", "bloque": 1, "parte": 1},
"2.1.1": {"nombre": "Acceso Vascular Básico", "bloque": 4, "parte": 2},
"2.1.2": {"nombre": "Reconocimiento PCR", "bloque": 4, "parte": 2},
"2.1.3": {"nombre": "RCP Adultos", "bloque": 4, "parte": 2},
"2.1.4": {"nombre": "RCP Pediatría", "bloque": 4, "parte": 2},
"2.1.5": {"nombre": "RCP Lactantes", "bloque": 4, "parte": 2},
"2.1.6": {"nombre": "Uso DESA", "bloque": 4, "parte": 2},
"2.1.7": {"nombre": "RCP Dos Intervinientes", "bloque": 4, "parte": 2},
"2.1.8": {"nombre": "OVACE Adultos", "bloque": 4, "parte": 2},
"2.1.9": {"nombre": "OVACE Pediatría", "bloque": 4, "parte": 2},
"2.1.10": {"nombre": "OVACE Lactantes", "bloque": 4, "parte": 2},
"2.1.11": {"nombre": "Posición Lateral de Seguridad", "bloque": 4, "parte": 2},
"2.2.1": {"nombre": "Medicina de Emergencias Aplicada", "bloque": 9, "parte": 2},
"3.1.1": {"nombre": "Anatomía Operativa", "bloque": 2, "parte": 3},
"3.1.2": {"nombre": "Inmovilización Manual", "bloque": 2, "parte": 3},
"3.1.3": {"nombre": "Collarín Cervical", "bloque": 2, "parte": 3},
"3.1.4": {"nombre": "Camilla Cuchara", "bloque": 2, "parte": 3},
"3.1.5": {"nombre": "Tablero Espinal", "bloque": 2, "parte": 3},
"3.1.6": {"nombre": "Colchón Vacío", "bloque": 2, "parte": 3},
"3.1.7": {"nombre": "Extricación y Movimientos en Bloque", "bloque": 2, "parte": 3},
"3.1.8": {"nombre": "Transferencias y Movilización", "bloque": 2, "parte": 3},
"3.1.9": {"nombre": "Errores Críticos", "bloque": 2, "parte": 3},
"3.1.10": {"nombre": "Férulas", "bloque": 2, "parte": 3},
"3.1.11": {"nombre": "Cinturón Pélvico", "bloque": 2, "parte": 3},
"3.1.12": {"nombre": "Férula de Tracción", "bloque": 2, "parte": 3},
"3.1.13": {"nombre": "Camillas y Sillas de Evacuación", "bloque": 2, "parte": 3},
"3.1.14": {"nombre": "Inventario de Material", "bloque": 2, "parte": 3},
"3.2.1": {"nombre": "Oxigenoterapia Básica", "bloque": 3, "parte": 3},
"3.2.2": {"nombre": "Oxigenoterapia - Fundamentos", "bloque": 3, "parte": 3},
"3.2.3": {"nombre": "Dispositivos de Oxigenoterapia", "bloque": 3, "parte": 3},
"3.2.4": {"nombre": "Ventilación con Bolsa-Mascarilla", "bloque": 3, "parte": 3},
"3.2.5": {"nombre": "Aspiración", "bloque": 3, "parte": 3},
"3.2.6": {"nombre": "Cánula Orofaringea", "bloque": 3, "parte": 3},
"3.2.7": {"nombre": "BVM (Bolsa Válvula Mascarilla)", "bloque": 3, "parte": 3},
"3.2.8": {"nombre": "Cánulas", "bloque": 3, "parte": 3},
"3.2.9": {"nombre": "Organización del Maletín", "bloque": 3, "parte": 3},
"3.2.10": {"nombre": "Control de Hemorragias", "bloque": 3, "parte": 3},
"3.2.11": {"nombre": "Quemaduras", "bloque": 3, "parte": 3},
"3.2.12": {"nombre": "Heridas y Vendajes", "bloque": 3, "parte": 3},
"3.2.13": {"nombre": "Exposición y Aislamiento Térmico", "bloque": 3, "parte": 3},
"3.2.14": {"nombre": "Monitorización Básica", "bloque": 3, "parte": 3},
"3.2.15": {"nombre": "Glucometro", "bloque": 3, "parte": 3},
"3.2.16": {"nombre": "Termometría", "bloque": 3, "parte": 3},
"3.2.17": {"nombre": "Confort y Dolor", "bloque": 3, "parte": 3},
"3.2.18": {"nombre": "Bioseguridad y Descontaminación", "bloque": 3, "parte": 3},
"3.2.19": {"nombre": "Gestión de Material en Escena", "bloque": 3, "parte": 3},
"3.2.20": {"nombre": "Comunicación Operativa", "bloque": 3, "parte": 3},
"3.2.21": {"nombre": "Señalización e Iluminación", "bloque": 3, "parte": 3},
"3.2.22": {"nombre": "Documentación Operativa", "bloque": 3, "parte": 3},
"3.2.23": {"nombre": "Cierre Bloque 3", "bloque": 3, "parte": 3},
"3.2.24": {"nombre": "Inventario Material Sanitario", "bloque": 3, "parte": 3},
"3.2.25": {"nombre": "Maletín de Curas", "bloque": 3, "parte": 3},
"3.2.26": {"nombre": "Bolsa de Monitorización", "bloque": 3, "parte": 3},
"3.2.27": {"nombre": "Inventario Global", "bloque": 3, "parte": 3},
"3.2.28": {"nombre": "Checklist Maestro", "bloque": 3, "parte": 3},
"4.1.1": {"nombre": "Principios de Administración de Fármacos", "bloque": 6, "parte": 4},
"4.1.2": {"nombre": "Vademécum Operativo", "bloque": 6, "parte": 4},
"4.1.3": {"nombre": "Oxígeno - Administración y Seguridad", "bloque": 6, "parte": 4},
"4.1.4": {"nombre": "Adrenalina - Uso en Anafilaxia y RCP", "bloque": 6, "parte": 4},
"4.1.5": {"nombre": "Aspirina - Uso en SCA", "bloque": 6, "parte": 4},
"4.1.6": {"nombre": "Glucagón - Uso en Hipoglucemia", "bloque": 6, "parte": 4},
"4.1.7": {"nombre": "Salbutamol - Uso en Crisis Asmática", "bloque": 6, "parte": 4},
"4.1.8": {"nombre": "Abreviaturas y Terminología Farmacológica", "bloque": 6, "parte": 4},
"5.1.1": {"nombre": "Introducción a Protocolos Transtelefónicos", "bloque": 5, "parte": 5},
"5.1.2": {"nombre": "Protocolos de Emergencias Específicas", "bloque": 5, "parte": 5},
"5.1.3": {"nombre": "PCR Transtelefónica", "bloque": 5, "parte": 5},
"5.1.4": {"nombre": "OVACE Transtelefónica", "bloque": 5, "parte": 5},
"5.1.5": {"nombre": "SCA Transtelefónico", "bloque": 5, "parte": 5},
"5.1.6": {"nombre": "ICTUS Transtelefónico", "bloque": 5, "parte": 5},
"5.1.7": {"nombre": "Anafilaxia Transtelefónica", "bloque": 5, "parte": 5},
"5.1.8": {"nombre": "Crisis Asmática Transtelefónica", "bloque": 5, "parte": 5},
"5.1.9": {"nombre": "Hipoglucemia Transtelefónica", "bloque": 5, "parte": 5},
"5.1.10": {"nombre": "Comunicación con Coordinador", "bloque": 5, "parte": 5},
"5.2.1": {"nombre": "Introducción a Gestión Operativa", "bloque": 8, "parte": 5},
"5.2.2": {"nombre": "Documentación Clínica Prehospitalaria", "bloque": 8, "parte": 5},
"5.2.3": {"nombre": "Coordinación y Comunicación Operativa", "bloque": 8, "parte": 5},
"5.2.4": {"nombre": "Gestión de Recursos y Material", "bloque": 8, "parte": 5},
"5.2.5": {"nombre": "Calidad y Mejora Continua", "bloque": 8, "parte": 5},
"6.1.1": {"nombre": "Fundamentos de Conducción en Urgencias", "bloque": 7, "parte": 6},
"6.1.2": {"nombre": "Uso de Luces y Sirena", "bloque": 7, "parte": 6},
"6.1.3": {"nombre": "Técnicas de Conducción en Emergencias", "bloque": 7, "parte": 6},
"6.1.4": {"nombre": "Seguridad Vial y Prevención de Accidentes", "bloque": 7, "parte": 6},
"6.1.5": {"nombre": "Gestión de Rutas y Navegación", "bloque": 7, "parte": 6},
"6.1.6": {"nombre": "Protocolos de Seguridad en Escena", "bloque": 7, "parte": 6},
"7.1.1": {"nombre": "Situaciones Especiales", "bloque": 10, "parte": 7},
"7.2.1": {"nombre": "Protocolos de Trauma", "bloque": 11, "parte": 7},
"8.1.1": {"nombre": "Marco Legal, Ético y Profesional del TES", "bloque": 12, "parte": 8},
"8.2.1": {"nombre": "Comunicación y Relación con el Paciente", "bloque": 13, "parte": 8},
"8.3.1": {"nombre": "Seguridad Personal y Salud del TES", "bloque": 14, "parte": 8},
}
MAPEO_REAL = {
"1.1.1": "BLOQUE_00_0_FUNDAMENTOS_EMERGENCIAS.md",
"1.2.1": "BLOQUE_01_1_CONSTANTES_VITALES.md",
"1.2.2": "BLOQUE_01_2_ABCDE_OPERATIVO.md",
"1.2.3": "BLOQUE_01_3_GLASGOW_OPERATIVO.md",
"1.2.4": "BLOQUE_01_4_TRIAGE_START.md",
"2.1.1": "BLOQUE_04_0_ACCESO_VASCULAR_BASICO.md",
"2.1.2": "BLOQUE_04_0B_RECONOCIMIENTO_PCR.md",
"2.1.3": "BLOQUE_04_1_RCP_ADULTOS.md",
"2.1.4": "BLOQUE_04_2_RCP_PEDIATRIA.md",
"2.1.5": "BLOQUE_04_3_RCP_LACTANTES.md",
"2.1.6": "BLOQUE_04_4_USO_DESA.md",
"2.1.7": "BLOQUE_04_5_RCP_DOS_INTERVINIENTES.md",
"2.1.8": "BLOQUE_04_6_OVACE_ADULTOS.md",
"2.1.9": "BLOQUE_04_7_OVACE_PEDIATRIA.md",
"2.1.10": "BLOQUE_04_8_OVACE_LACTANTES.md",
"2.1.11": "BLOQUE_04_9_POSICION_LATERAL_SEGURIDAD.md",
"2.2.1": "BLOQUE_09_0_MEDICINA_EMERGENCIAS_APLICADA.md",
"3.1.1": "BLOQUE_02_0_ANATOMIA_OPERATIVA.md",
"3.1.2": "BLOQUE_02_2_INMOVILIZACION_MANUAL.md",
"3.1.3": "BLOQUE_02_3_COLLARIN_CERVICAL.md",
"3.1.4": "BLOQUE_02_4_CAMILLA_CUCHARA.md",
"3.1.5": "BLOQUE_02_5_TABLERO_ESPINAL.md",
"3.1.6": "BLOQUE_02_6_COLCHON_VACIO.md",
"3.1.7": "BLOQUE_02_7_EXTRICACION_MOVIMIENTOS_BLOQUE.md",
"3.1.8": "BLOQUE_02_8_TRANSFERENCIAS_MOVILIZACION.md",
"3.1.9": "BLOQUE_02_9_ERRORES_CRITICOS.md",
"3.1.10": "BLOQUE_02_10_FERULAS.md",
"3.1.11": "BLOQUE_02_11_CINTURON_PELVICO.md",
"3.1.12": "BLOQUE_02_12_FERULA_TRACCION.md",
"3.1.13": "BLOQUE_02_13_CAMILLAS_SILLAS_EVACUACION.md",
"3.1.14": "BLOQUE_02_X_INVENTARIO_MATERIAL.md",
"3.2.1": "BLOQUE_03_0A_OXIGENOTERAPIA_BASICA.md",
"3.2.2": "BLOQUE_03_0B_OXIGENOTERAPIA_FUNDAMENTOS.md",
"3.2.3": "BLOQUE_03_1_DISPOSITIVOS_OXIGENOTERAPIA.md",
"3.2.4": "BLOQUE_03_1B_VENTILACION_BOLSA_MASCARILLA.md",
"3.2.5": "BLOQUE_03_2_ASPIRACION.md",
"3.2.6": "BLOQUE_03_2B_CANULA_OROFARINGEA.md",
"3.2.7": "BLOQUE_03_3_BVM.md",
"3.2.8": "BLOQUE_03_4_CANULAS.md",
"3.2.9": "BLOQUE_03_5_ORGANIZACION_MALETIN.md",
"3.2.10": "BLOQUE_03_6_CONTROL_HEMORRAGIAS.md",
"3.2.11": "BLOQUE_03_7_QUEMADURAS.md",
"3.2.12": "BLOQUE_03_8_HERIDAS_VENDAJES.md",
"3.2.13": "BLOQUE_03_9_EXPOSICION_AISLAMIENTO_TERMICO.md",
"3.2.14": "BLOQUE_03_10_MONITORIZACION_BASICA.md",
"3.2.15": "BLOQUE_03_11_GLUCOMETRO.md",
"3.2.16": "BLOQUE_03_12_TERMOMETRIA.md",
"3.2.17": "BLOQUE_03_13_CONFORT_DOLOR.md",
"3.2.18": "BLOQUE_03_14_BIOSEGURIDAD_DESCONTAMINACION.md",
"3.2.19": "BLOQUE_03_15_GESTION_MATERIAL_ESCENA.md",
"3.2.20": "BLOQUE_03_16_COMUNICACION_OPERATIVA.md",
"3.2.21": "BLOQUE_03_17_SENALIZACION_ILUMINACION.md",
"3.2.22": "BLOQUE_03_18_DOCUMENTACION_OPERATIVA.md",
"3.2.23": "BLOQUE_03_99_CIERRE_BLOQUE_3.md",
"3.2.24": "BLOQUE_03_X_INVENTARIO_MATERIAL_SANITARIO.md",
"3.2.25": "BLOQUE_03_X2_MALETIN_CURAS.md",
"3.2.26": "BLOQUE_03_X3_BOLSA_MONITORIZACION.md",
"3.2.27": "BLOQUE_03_X4_INVENTARIO_GLOBAL.md",
"3.2.28": "BLOQUE_03_X5_CHECKLIST_MAESTRO.md",
"4.1.1": "BLOQUE_06_0_PRINCIPIOS_ADMINISTRACION_FARMACOS.md",
"4.1.2": "BLOQUE_06_1_VADEMECUM_OPERATIVO.md",
"4.1.3": "BLOQUE_06_2_OXIGENO_ADMINISTRACION_Y_SEGURIDAD.md",
"4.1.4": "BLOQUE_06_3_ADRENALINA_USO_ANAFILAXIA_Y_RCP.md",
"4.1.5": "BLOQUE_06_4_ASPIRINA_USO_SCA.md",
"4.1.6": "BLOQUE_06_5_GLUCAGON_USO_HIPOGLUCEMIA.md",
"4.1.7": "BLOQUE_06_6_SALBUTAMOL_USO_CRISIS_ASMATICA.md",
"4.1.8": "BLOQUE_06_7_ABREVIATURAS_TERMINOLOGIA_FARMACOLOGICA.md",
"5.1.1": "BLOQUE_05_0_INTRODUCCION_PROTOCOLOS_TRANSTELEFONICOS.md",
"5.1.2": "BLOQUE_05_0B_PROTOCOLOS_EMERGENCIAS_ESPECIFICAS.md",
"5.1.3": "BLOQUE_05_1_PCR_TRANSTELEFONICA.md",
"5.1.4": "BLOQUE_05_2_OVACE_TRANSTELEFONICA.md",
"5.1.5": "BLOQUE_05_3_SCA_TRANSTELEFONICO.md",
"5.1.6": "BLOQUE_05_4_ICTUS_TRANSTELEFONICO.md",
"5.1.7": "BLOQUE_05_5_ANAFILAXIA_TRANSTELEFONICA.md",
"5.1.8": "BLOQUE_05_6_CRISIS_ASMATICA_TRANSTELEFONICA.md",
"5.1.9": "BLOQUE_05_7_HIPOGLUCEMIA_TRANSTELEFONICA.md",
"5.1.10": "BLOQUE_05_8_COMUNICACION_COORDINADOR.md",
"5.2.1": "BLOQUE_08_0_INTRODUCCION_GESTION_OPERATIVA.md",
"5.2.2": "BLOQUE_08_1_DOCUMENTACION_CLINICA_PREHOSPITALARIA.md",
"5.2.3": "BLOQUE_08_2_COORDINACION_Y_COMUNICACION_OPERATIVA.md",
"5.2.4": "BLOQUE_08_3_GESTION_RECURSOS_Y_MATERIAL.md",
"5.2.5": "BLOQUE_08_4_CALIDAD_Y_MEJORA_CONTINUA.md",
"6.1.1": "BLOQUE_07_0_FUNDAMENTOS_CONDUCCION_URGENCIAS.md",
"6.1.2": "BLOQUE_07_1_USO_LUCES_Y_SIRENA.md",
"6.1.3": "BLOQUE_07_2_TECNICAS_CONDUCCION_EMERGENCIAS.md",
"6.1.4": "BLOQUE_07_3_SEGURIDAD_VIAL_Y_PREVENCION_ACCIDENTES.md",
"6.1.5": "BLOQUE_07_4_GESTION_RUTAS_Y_NAVEGACION.md",
"6.1.6": "BLOQUE_07_5_PROTOCOLOS_SEGURIDAD_EN_ESCENA.md",
"7.1.1": "BLOQUE_10_0_SITUACIONES_ESPECIALES.md",
"7.2.1": "BLOQUE_11_0_PROTOCOLOS_TRAUMA.md",
"8.1.1": "BLOQUE_12_0_MARCO_LEGAL_ETICO_PROFESIONAL.md",
"8.2.1": "BLOQUE_13_0_COMUNICACION_RELACION_PACIENTE.md",
"8.3.1": "BLOQUE_14_0_SEGURIDAD_PERSONAL_SALUD_TES.md",
}
BLOQUE_TO_FOLDER = {
0: "BLOQUE_0_FUNDAMENTOS",
1: "BLOQUE_1_PROCEDIMIENTOS_BASICOS",
2: "BLOQUE_2_MATERIAL_E_INMOVILIZACION",
3: "BLOQUE_3_MATERIAL_SANITARIO_Y_OXIGENOTERAPIA",
4: "BLOQUE_4_SOPORTE_VITAL_BASICO_Y_RCP",
5: "BLOQUE_5_PROTOCOLOS_TRANSTELEFONICOS",
6: "BLOQUE_6_FARMACOLOGIA",
7: "BLOQUE_7_CONDUCCION_Y_SEGURIDAD_VIAL",
8: "BLOQUE_8_GESTION_OPERATIVA_Y_DOCUMENTACION",
9: "BLOQUE_9_MEDICINA_EMERGENCIAS_APLICADA",
10: "BLOQUE_10_SITUACIONES_ESPECIALES",
11: "BLOQUE_11_PROTOCOLOS_TRAUMA",
12: "BLOQUE_12_MARCO_LEGAL_ETICO_PROFESIONAL",
13: "BLOQUE_13_COMUNICACION_RELACION_PACIENTE",
14: "BLOQUE_14_SEGURIDAD_PERSONAL_SALUD_TES",
}
PARTES_INFO = {
1: {"nombre": "Fundamentos y Evaluación Inicial", "icono": "BookOpen"},
2: {"nombre": "Soporte Vital y Procedimientos Críticos", "icono": "Heart"},
3: {"nombre": "Material y Equipamiento", "icono": "Package"},
4: {"nombre": "Farmacología y Medicamentos", "icono": "Pill"},
5: {"nombre": "Protocolos y Gestión Operativa", "icono": "Phone"},
6: {"nombre": "Conducción y Seguridad Vial", "icono": "Car"},
7: {"nombre": "Situaciones Especiales y Trauma", "icono": "AlertTriangle"},
8: {"nombre": "Habilidades Profesionales", "icono": "User"},
}
BLOQUES_INFO = {
0: {"nombre": "Fundamentos de Emergencias Prehospitalarias", "icono": "BookOpen"},
1: {"nombre": "Procedimientos Básicos", "icono": "ClipboardCheck"},
2: {"nombre": "Material e Inmovilización", "icono": "Package"},
3: {"nombre": "Material Sanitario y Oxigenoterapia", "icono": "Wind"},
4: {"nombre": "Soporte Vital Básico y RCP", "icono": "Heart"},
5: {"nombre": "Protocolos Transtelefónicos", "icono": "Phone"},
6: {"nombre": "Farmacología y Vademécum Operativo", "icono": "Pill"},
7: {"nombre": "Conducción y Seguridad Vial", "icono": "Car"},
8: {"nombre": "Gestión Operativa y Documentación", "icono": "FileText"},
9: {"nombre": "Medicina de Emergencias Aplicada", "icono": "Stethoscope"},
10: {"nombre": "Situaciones Especiales", "icono": "AlertTriangle"},
11: {"nombre": "Protocolos de Trauma", "icono": "Activity"},
12: {"nombre": "Marco Legal, Ético y Profesional", "icono": "Scale"},
13: {"nombre": "Comunicación y Relación con el Paciente", "icono": "MessageSquare"},
14: {"nombre": "Seguridad Personal y Salud del TES", "icono": "Shield"},
}
def generar_ruta_url(codigo: str, parte: int, bloque: int) -> str:
"""Genera ruta URL amigable"""
parte_nombres = {
1: "parte-i-fundamentos",
2: "parte-ii-soporte-vital",
3: "parte-iii-material",
4: "parte-iv-farmacologia",
5: "parte-v-protocolos",
6: "parte-vi-conduccion",
7: "parte-vii-situaciones-especiales",
8: "parte-viii-habilidades",
}
bloque_nombres = {
0: "bloque-0-fundamentos",
1: "bloque-1-procedimientos-basicos",
2: "bloque-2-inmovilizacion",
3: "bloque-3-oxigenoterapia",
4: "bloque-4-rcp",
5: "bloque-5-transtelefonicos",
6: "bloque-6-farmacologia",
7: "bloque-7-conduccion",
8: "bloque-8-gestion",
9: "bloque-9-medicina-emergencias",
10: "bloque-10-situaciones-especiales",
11: "bloque-11-trauma",
12: "bloque-12-marco-legal",
13: "bloque-13-comunicacion",
14: "bloque-14-seguridad",
}
return f"/manual/{parte_nombres[parte]}/{bloque_nombres[bloque]}/{codigo}"
def determinar_nivel_dificultad(codigo: str, bloque: int) -> str:
"""Determina nivel de dificultad basado en bloque y código"""
if bloque == 0 or codigo.startswith("1."):
return "basico"
elif bloque in [4, 5] or codigo.startswith("2."):
return "avanzado"
else:
return "intermedio"
def determinar_importancia(codigo: str, bloque: int) -> str:
"""Determina importancia basado en contenido"""
protocolos_criticos = ["2.1.3", "2.1.8", "2.1.6", "1.2.2"] # RCP, OVACE, DESA, ABCDE
if codigo in protocolos_criticos or bloque == 4:
return "alta"
elif bloque in [0, 1, 2, 3]:
return "alta"
else:
return "media"
def extraer_palabras_clave(nombre: str, bloque: int) -> list:
"""Extrae palabras clave del nombre"""
palabras = nombre.lower().split()
palabras_clave = []
# Agregar palabras relevantes
terminos_importantes = [
"rcp", "ovace", "abcde", "glasgow", "triage", "oxigenoterapia",
"farmacologia", "trauma", "inmovilizacion", "protocolo", "emergencia"
]
for termino in terminos_importantes:
if termino in nombre.lower():
palabras_clave.append(termino)
return palabras_clave[:5] # Máximo 5 palabras clave
def calcular_tiempo_lectura(ruta_archivo: Path) -> int:
"""Calcula tiempo estimado de lectura"""
try:
with open(ruta_archivo, 'r', encoding='utf-8') as f:
contenido = f.read()
palabras = len(contenido.split())
# Promedio: 200 palabras por minuto
return max(5, palabras // 200)
except:
return 10 # Default
def obtener_capitulos_relacionados(codigo: str, parte: int, bloque: int) -> list:
"""Obtiene IDs de capítulos relacionados"""
relacionados = []
# Misma parte
for cod, info in INDICE_ESPERADO.items():
if info["parte"] == parte and cod != codigo:
relacionados.append(cod)
# Limitar a 5 relacionados
return relacionados[:5]
def generar_indice():
"""Genera el archivo manual-index.ts"""
# Organizar por partes y bloques
partes_estructura = {}
for codigo, info in INDICE_ESPERADO.items():
parte_num = info["parte"]
bloque_num = info["bloque"]
if parte_num not in partes_estructura:
partes_estructura[parte_num] = {
"bloques": {}
}
if bloque_num not in partes_estructura[parte_num]["bloques"]:
partes_estructura[parte_num]["bloques"][bloque_num] = {
"capitulos": []
}
# Obtener archivo real
nombre_archivo = MAPEO_REAL.get(codigo)
if nombre_archivo:
folder_name = BLOQUE_TO_FOLDER[bloque_num]
ruta_archivo = MANUAL_DIR / folder_name / nombre_archivo
# Calcular navegación
codigos_ordenados = sorted(INDICE_ESPERADO.keys())
indice_actual = codigos_ordenados.index(codigo)
anterior = codigos_ordenados[indice_actual - 1] if indice_actual > 0 else None
siguiente = codigos_ordenados[indice_actual + 1] if indice_actual < len(codigos_ordenados) - 1 else None
capitulo = {
"id": codigo,
"titulo": info["nombre"],
"parte": parte_num,
"parteNombre": PARTES_INFO[parte_num]["nombre"],
"bloque": bloque_num,
"bloqueNombre": BLOQUES_INFO[bloque_num]["nombre"],
"rutaArchivo": f"manual-tes/TES_Manual_Digital/{folder_name}/{nombre_archivo}",
"rutaUrl": generar_ruta_url(codigo, parte_num, bloque_num),
"nivelDificultad": determinar_nivel_dificultad(codigo, bloque_num),
"importancia": determinar_importancia(codigo, bloque_num),
"palabrasClave": extraer_palabras_clave(info["nombre"], bloque_num),
"tipoContenido": "operativo" if bloque_num in [1, 2, 3, 4] else "formativo",
"tiempoLectura": calcular_tiempo_lectura(ruta_archivo),
"navegacion": {
"anterior": anterior,
"siguiente": siguiente,
"relacionados": obtener_capitulos_relacionados(codigo, parte_num, bloque_num)
},
"metadata": {
"version": "1.0",
"fechaActualizacion": "2024-12-13",
"autor": "Manual TES Digital"
}
}
partes_estructura[parte_num]["bloques"][bloque_num]["capitulos"].append(capitulo)
# Generar código TypeScript
ts_code = []
ts_code.append("// Índice del Manual TES Digital")
ts_code.append("// Generado automáticamente desde estructura de archivos")
ts_code.append("// Fecha: 2025-12-17")
ts_code.append("")
ts_code.append("export interface Capitulo {")
ts_code.append(" id: string;")
ts_code.append(" titulo: string;")
ts_code.append(" parte: number;")
ts_code.append(" parteNombre: string;")
ts_code.append(" bloque: number;")
ts_code.append(" bloqueNombre: string;")
ts_code.append(" rutaArchivo: string;")
ts_code.append(" rutaUrl: string;")
ts_code.append(" nivelDificultad: 'basico' | 'intermedio' | 'avanzado';")
ts_code.append(" importancia: 'alta' | 'media' | 'baja';")
ts_code.append(" palabrasClave: string[];")
ts_code.append(" tipoContenido: 'formativo' | 'operativo' | 'referencia';")
ts_code.append(" tiempoLectura: number;")
ts_code.append(" navegacion: {")
ts_code.append(" anterior: string | null;")
ts_code.append(" siguiente: string | null;")
ts_code.append(" relacionados: string[];")
ts_code.append(" };")
ts_code.append(" metadata: {")
ts_code.append(" version: string;")
ts_code.append(" fechaActualizacion: string;")
ts_code.append(" autor: string;")
ts_code.append(" };")
ts_code.append("}")
ts_code.append("")
ts_code.append("export interface Bloque {")
ts_code.append(" id: number;")
ts_code.append(" nombre: string;")
ts_code.append(" descripcion: string;")
ts_code.append(" icono?: string;")
ts_code.append(" capitulos: Capitulo[];")
ts_code.append("}")
ts_code.append("")
ts_code.append("export interface Parte {")
ts_code.append(" id: number;")
ts_code.append(" nombre: string;")
ts_code.append(" descripcion: string;")
ts_code.append(" icono?: string;")
ts_code.append(" bloques: Bloque[];")
ts_code.append("}")
ts_code.append("")
ts_code.append("export const manualIndex: Parte[] = [")
# Generar estructura de partes
for parte_num in sorted(partes_estructura.keys()):
parte_info = PARTES_INFO[parte_num]
ts_code.append(f" {{")
ts_code.append(f" id: {parte_num},")
ts_code.append(f" nombre: \"{parte_info['nombre']}\",")
ts_code.append(f" descripcion: \"\",")
ts_code.append(f" icono: \"{parte_info['icono']}\",")
ts_code.append(f" bloques: [")
# Generar bloques
for bloque_num in sorted(partes_estructura[parte_num]["bloques"].keys()):
bloque_info = BLOQUES_INFO[bloque_num]
capitulos = sorted(partes_estructura[parte_num]["bloques"][bloque_num]["capitulos"],
key=lambda x: x["id"])
ts_code.append(f" {{")
ts_code.append(f" id: {bloque_num},")
ts_code.append(f" nombre: \"{bloque_info['nombre']}\",")
ts_code.append(f" descripcion: \"\",")
ts_code.append(f" icono: \"{bloque_info['icono']}\",")
ts_code.append(f" capitulos: [")
# Generar capítulos
for cap in capitulos:
ts_code.append(f" {{")
ts_code.append(f" id: \"{cap['id']}\",")
ts_code.append(f" titulo: \"{cap['titulo']}\",")
ts_code.append(f" parte: {cap['parte']},")
ts_code.append(f" parteNombre: \"{cap['parteNombre']}\",")
ts_code.append(f" bloque: {cap['bloque']},")
ts_code.append(f" bloqueNombre: \"{cap['bloqueNombre']}\",")
ts_code.append(f" rutaArchivo: \"{cap['rutaArchivo']}\",")
ts_code.append(f" rutaUrl: \"{cap['rutaUrl']}\",")
ts_code.append(f" nivelDificultad: \"{cap['nivelDificultad']}\",")
ts_code.append(f" importancia: \"{cap['importancia']}\",")
ts_code.append(f" palabrasClave: {json.dumps(cap['palabrasClave'])},")
ts_code.append(f" tipoContenido: \"{cap['tipoContenido']}\",")
ts_code.append(f" tiempoLectura: {cap['tiempoLectura']},")
ts_code.append(f" navegacion: {{")
ts_code.append(f" anterior: {json.dumps(cap['navegacion']['anterior'])},",)
ts_code.append(f" siguiente: {json.dumps(cap['navegacion']['siguiente'])},",)
ts_code.append(f" relacionados: {json.dumps(cap['navegacion']['relacionados'])}")
ts_code.append(f" }},")
ts_code.append(f" metadata: {{")
ts_code.append(f" version: \"{cap['metadata']['version']}\",")
ts_code.append(f" fechaActualizacion: \"{cap['metadata']['fechaActualizacion']}\",")
ts_code.append(f" autor: \"{cap['metadata']['autor']}\"")
ts_code.append(f" }}")
ts_code.append(f" }},")
ts_code.append(f" ]")
ts_code.append(f" }},")
ts_code.append(f" ]")
ts_code.append(f" }},")
ts_code.append("];")
ts_code.append("")
ts_code.append("// Función helper para obtener capítulo por ID")
ts_code.append("export function getCapituloById(id: string): Capitulo | null {")
ts_code.append(" for (const parte of manualIndex) {")
ts_code.append(" for (const bloque of parte.bloques) {")
ts_code.append(" const capitulo = bloque.capitulos.find(c => c.id === id);")
ts_code.append(" if (capitulo) return capitulo;")
ts_code.append(" }")
ts_code.append(" }")
ts_code.append(" return null;")
ts_code.append("}")
ts_code.append("")
ts_code.append("// Función helper para obtener todos los capítulos")
ts_code.append("export function getAllCapitulos(): Capitulo[] {")
ts_code.append(" const capitulos: Capitulo[] = [];")
ts_code.append(" for (const parte of manualIndex) {")
ts_code.append(" for (const bloque of parte.bloques) {")
ts_code.append(" capitulos.push(...bloque.capitulos);")
ts_code.append(" }")
ts_code.append(" }")
ts_code.append(" return capitulos;")
ts_code.append("}")
# Guardar archivo
output_path = BASE_DIR / "src" / "data" / "manual-index.ts"
with open(output_path, 'w', encoding='utf-8') as f:
f.write('\n'.join(ts_code))
print(f"✅ Índice generado: {output_path}")
print(f" Total de capítulos: {len(INDICE_ESPERADO)}")
if __name__ == "__main__":
generar_indice()