LEGALFAMI / app.py
km1lo's picture
Upload app.py
2d161de verified
# -*- coding: utf-8 -*-
"""app.ipynb
Automatically generated by Colab.
Original file is located at
https://colab.research.google.com/drive/1e3y3OTjQnNFZ7FLgs2elviYcG7ptj2fI
"""
from transformers import AutoTokenizer, AutoModelForCausalLM, pipeline
from sentence_transformers import SentenceTransformer, util
import numpy as np
import joblib
import torch
import gradio as gr
from transformers import TextGenerationPipeline
# === Cargar modelos entrenados ===
modelo_riesgo = joblib.load("modelo_riesgo.pkl")
modelo_violencia = joblib.load("modelo_tipo_violencia.pkl")
modelo_medida = joblib.load("modelo_tipo_medida.pkl")
codificadores = joblib.load("codificadores_label_encoder.pkl")
modelo_vector = SentenceTransformer("Snowflake/snowflake-arctic-embed-xs")
# === Cargar modelo de lenguaje OpenHermes-2.5 ===
model_id = "teknium/OpenHermes-2.5-Mistral-7B"
tokenizer = AutoTokenizer.from_pretrained(model_id, trust_remote_code=True)
model = AutoModelForCausalLM.from_pretrained(
model_id,
trust_remote_code=True,
torch_dtype=torch.float16 if torch.cuda.is_available() else torch.float32,
device_map="auto"
)
modelo_llm = TextGenerationPipeline(
model=model,
tokenizer=tokenizer,
max_new_tokens=300,
temperature=0.5,
do_sample=True,
)
# === Frases prototipo para verificador semántico ===
frases_fisica = [
"Me golpeó con el puño cerrado", "Me pateó", "Me lanzó contra la pared",
"Me estranguló", "Me fracturó una costilla", "Me tiró al piso violentamente"
]
frases_sexual = [
"Me obligó a tener relaciones sexuales", "Me tocó sin consentimiento",
"Me violó", "Me forzó a tener sexo", "Me agredió sexualmente"
]
embeds_fisica = modelo_vector.encode(frases_fisica)
embeds_sexual = modelo_vector.encode(frases_sexual)
# === FUNCIÓN PRINCIPAL ===
def predecir_con_recomendacion(edad, genero, hijos, convivencia_agresor, consumo_sustancias, apoyo_familiar, descripcion):
# Codificar variables
vector_tabular = np.array([
int(edad),
int(hijos),
codificadores["genero"].transform([genero])[0],
0, 0, 0,
codificadores["convivencia_agresor"].transform([convivencia_agresor])[0],
codificadores["consumo_sustancias"].transform([consumo_sustancias])[0],
codificadores["apoyo_familiar"].transform([apoyo_familiar])[0]
])
# Vectorizar descripción
vector_desc = modelo_vector.encode([descripcion])[0]
entrada = np.concatenate([vector_tabular, vector_desc])
# Predicciones
riesgo_cod = modelo_riesgo.predict([entrada])[0]
tipo_violencia_cod = modelo_violencia.predict([entrada])[0]
tipo_medida_cod = modelo_medida.predict([entrada])[0]
# Decodificar
riesgo = codificadores["riesgo"].inverse_transform([riesgo_cod])[0]
tipo_violencia = codificadores["tipo_violencia"].inverse_transform([tipo_violencia_cod])[0]
tipo_medida = codificadores["tipo_medida"].inverse_transform([tipo_medida_cod])[0]
# Verificador semántico
emb_desc = modelo_vector.encode(descripcion)
sim_fisica = max(util.cos_sim(emb_desc, embeds_fisica)[0])
sim_sexual = max(util.cos_sim(emb_desc, embeds_sexual)[0])
if sim_fisica > 0.7:
tipo_violencia = "física"
elif sim_sexual > 0.7:
tipo_violencia = "sexual"
# PROMPT legal
prompt = f"""[INSTRUCCIÓN]
Como asistente legal, tu tarea es determinar qué medida de protección debe aplicarse en un caso de violencia intrafamiliar, de acuerdo al Artículo 5 de la Ley 575 de 2000 (modificado por la Ley 1257 de 2008 y la Ley 2126 de 2021).
[CASO]
Tipo de violencia: {tipo_violencia}
Nivel de riesgo: {riesgo}
Tipo de medida cautelar: {tipo_medida}
Descripción del caso: {descripcion}
[OBJETIVO]
Con base en la descripción del caso y la ley, indica:
- Qué literal(es) del Artículo 5 (a-n) deben aplicarse.
- Justifica legalmente tu elección con argumentos claros y precisos.
[RESPUESTA]
"""
respuesta = modelo_llm(prompt, max_new_tokens=350, temperature=0.5)[0]["generated_text"]
razonamiento = respuesta.split("[RESPUESTA]")[-1].strip()
return tipo_violencia, riesgo, tipo_medida, razonamiento
# === Gradio UI ===
gr.Interface(
fn=predecir_con_recomendacion,
inputs=[
gr.Slider(18, 65, value=30, label="Edad"),
gr.Radio(["F", "M"], label="Género"),
gr.Slider(0, 5, step=1, value=1, label="Número de hijos"),
gr.Radio(["sí", "no"], label="¿Convive con el agresor?"),
gr.Radio(["sí", "no"], label="¿Hay consumo de sustancias?"),
gr.Radio(["sí", "no"], label="¿Tiene apoyo familiar?"),
gr.Textbox(lines=4, label="Descripción del caso")
],
outputs=[
gr.Textbox(label="🛑 Tipo de violencia predicho"),
gr.Textbox(label="⚠️ Nivel de riesgo"),
gr.Textbox(label="🧾 Tipo de medida cautelar"),
gr.Textbox(label="📜 Recomendación legal razonada")
],
title="LEGALFAMI – Asistente Legal con Razonamiento Jurídico",
description="Predice tipo de violencia, nivel de riesgo y medida cautelar según Ley 575 Artículo 5, y recomienda la acción legal correspondiente.",
theme="default"
).launch()