daqc's picture
Update app.py
d6e70b1 verified
import os
import gradio as gr
from gradio import ChatMessage
import torch
import torch._dynamo
from transformers import AutoModelForCausalLM, AutoTokenizer
from threading import Thread
from huggingface_hub import hf_hub_download, login
from dotenv import load_dotenv
import re
from llama_cpp import Llama
from typing import Iterator
# Cargar variables de entorno
load_dotenv()
# Configurar token de Hugging Face
HF_TOKEN = os.getenv("HF_TOKEN")
if HF_TOKEN:
login(token=HF_TOKEN)
# Intentar importar spaces solo si estamos en un espacio de Hugging Face
try:
import spaces
SPACES_AVAILABLE = True
except ImportError:
SPACES_AVAILABLE = False
# Desactivar TorchDynamo para evitar errores de compilación
torch._dynamo.config.suppress_errors = True
torch._dynamo.disable()
# Configuración
MODEL_ID = "somosnlp-hackathon-2025/iberotales-gemma-3-1b-it-es"
GGUF_MODEL_ID = "somosnlp-hackathon-2025/iberotales-gemma-3-1b-it-es-finetune-gguf"
GGUF_FILENAME = "gemma-3-finetune.Q8_0.gguf"
GGUF_REVISION = "main"
MAX_MAX_NEW_TOKENS = 2048
DEFAULT_MAX_NEW_TOKENS = 2048
# Verificar si estamos en un espacio de Hugging Face
IS_HF_SPACE = any([
os.getenv("SPACE_ID") is not None,
os.getenv("SPACE_AUTHOR_NAME") is not None,
os.getenv("SPACE_REPO_NAME") is not None,
os.getenv("SPACE_HOST") is not None,
])
# System prompt personalizado
DEFAULT_SYSTEM_MESSAGE = """Resuelve el siguiente problema.
Primero, piensa en voz alta qué debes hacer, paso por paso y de forma resumida, entre <think> y </think>.
Luego, da la respuesta final entre <SOLUTION> y </SOLUTION>.
No escribas nada fuera de ese formato."""
# Base de datos de personajes por país con banderas
PERSONAJES_POR_PAIS = {
"🇦🇷 Argentina": [
{"nombre": "La Difunta Correa", "imagen": "images/ar1.jpg", "descripcion": "Santa popular que murió de sed siguiendo a su esposo reclutado"},
{"nombre": "El Lobizón", "imagen": "images/ar2.jpg", "descripcion": "Hombre lobo de la tradición gaucha, séptimo hijo varón maldito"},
{"nombre": "La Telesita", "imagen": "images/ar3.webp", "descripcion": "Bailarina folklórica que se aparece en festivales y zambas"}
],
"🇧🇴 Bolivia": [
{"nombre": "El Tío del Cerro Rico", "imagen": "images/bo1.webp", "descripcion": "Señor de las minas que protege y castiga a los mineros"},
{"nombre": "El Ekeko", "imagen": "images/bo2.jpg", "descripcion": "Dios aymara de la abundancia y la fortuna con jorobas"},
{"nombre": "El Jichi", "imagen": "images/bo3.webp", "descripcion": "Serpiente protectora de ríos y lagunas en la cultura andina"}
],
"🇧🇷 Brasil": [
{"nombre": "Curupira", "imagen": "images/br1.jpeg", "descripcion": "Protector del bosque amazónico con pies al revés"},
{"nombre": "Saci-Pererê", "imagen": "images/br2.jpg", "descripcion": "Duende travieso de una pierna que fuma pipa"},
{"nombre": "Yebá Bëló", "imagen": "images/br3.jpg", "descripcion": "Abuela del mundo en mitología desana, creadora del universo"}
],
"🇨🇱 Chile": [
{"nombre": "Guallipén", "imagen": "images/ch1.webp", "descripcion": "Carnero gigante que habita en los ríos de Chiloé"},
{"nombre": "Colo Colo", "imagen": "images/ch2.webp", "descripcion": "Ser maléfico nacido de huevo de gallo empollado por serpiente"},
{"nombre": "Cuchivilu", "imagen": "images/ch3.webp", "descripcion": "Cerdo marino con cola de pez que habita en Chiloé"}
],
"🇨🇴 Colombia": [
{"nombre": "El Guaca", "imagen": "images/co1.jpg", "descripcion": "Espíritu guardián de tesoros enterrados por indígenas"},
{"nombre": "Chiminigagua", "imagen": "images/co2.webp", "descripcion": "Dios creador muisca, fuente de luz y vida universal"},
{"nombre": "Jukumari", "imagen": "images/co3.jpg", "descripcion": "Oso gigante mitológico de los Andes colombianos"}
],
"🇨🇷 Costa Rica": [
{"nombre": "El Micomalo", "imagen": "images/cr1.jpg", "descripcion": "Mono gigante y feroz que ataca a los cazadores"},
{"nombre": "La Carreta sin Bueyes", "imagen": "images/crr2.jpg", "descripcion": "Carreta fantasma que recorre caminos sin animales tirando"},
{"nombre": "El Sisimiqui", "imagen": "images/cr3.webp", "descripcion": "Duende pequeño y travieso de los bosques costarricenses"}
],
"🇨🇺 Cuba": [
{"nombre": "El Güije", "imagen": "images/cu1.jpg", "descripcion": "Duende negro de aguas dulces que ahoga a los bañistas"},
{"nombre": "Itiba Cahubaba", "imagen": "images/cu2.jpg", "descripcion": "Madre primordial taína de donde brotaron las aguas"},
{"nombre": "Guabancex", "imagen": "images/cu3.jpg", "descripcion": "Diosa taína de los huracanes y vientos destructores"}
],
"🇪🇨 Ecuador": [
{"nombre": "Salun", "imagen": "images/ec1.jpg", "descripcion": "Espíritu shamán shuar que guía en visiones espirituales"},
{"nombre": "Nunkui", "imagen": "images/ec2.jpg", "descripcion": "Diosa shuar de la fertilidad de la tierra y agricultura"},
{"nombre": "Yacuruna", "imagen": "images/ec3.jpg", "descripcion": "Hombre serpiente amazónico señor de las aguas dulces"}
],
"🇪🇸 España": [
{"nombre": "Ojáncanu", "imagen": "images/es1.jpg", "descripcion": "Gigante cíclope cántabro de fuerza descomunal"},
{"nombre": "Los Carantos", "imagen": "images/es2.jpg", "descripcion": "Espíritus gallegos que anuncian desgracias familiares"},
{"nombre": "Cuegle", "imagen": "images/es3.jpg", "descripcion": "Ser asturiano de tres ojos que protege casas del mal"}
],
"🇬🇹 Guatemala": [
{"nombre": "Camalotz", "imagen": "images/gu1.jpg", "descripcion": "Dios murciélago maya señor de las cuevas y la muerte"},
{"nombre": "La Siguanaba", "imagen": "images/gu2.jpg", "descripcion": "Mujer hermosa que muestra rostro de calavera a infieles"},
{"nombre": "Gucumatz", "imagen": "images/gu3.jpg", "descripcion": "Serpiente emplumada maya, equivalente a Quetzalcóatl"}
],
"🇭🇳 Honduras": [
{"nombre": "Icelaca", "imagen": "images/ho1.jpg", "descripcion": "Serpiente gigante que habita en cuevas hondureñas"},
{"nombre": "Cihuateteo", "imagen": "images/ho2.jpg", "descripcion": "Espíritus de mujeres muertas en parto, guerreras divinas"},
{"nombre": "Comelenguas", "imagen": "images/ho3.webp", "descripcion": "Demonio que devora lenguas de personas dormidas"}
],
"🇲🇽 México": [
{"nombre": "Quetzalcóatl", "imagen": "images/me1.jpg", "descripcion": "Serpiente emplumada, dios del viento y la sabiduría"},
{"nombre": "Tlacatecolotl", "imagen": "images/me2.jpg", "descripcion": "Brujo nahuatl que se transforma en búho gigante"},
{"nombre": "Tlalocan", "imagen": "images/me3.jpg", "descripcion": "Paraíso acuático de Tláloc donde van los ahogados"}
],
"🇳🇮 Nicaragua": [
{"nombre": "La Mocuana", "imagen": "images/ni1.webp", "descripcion": "Espíritu de mujer sin cabeza que vaga de noche"},
{"nombre": "El Guácimo Renco", "imagen": "images/ni2.jpg", "descripcion": "Árbol maldito que se transforma y persigue viajeros"},
{"nombre": "La Mona Bruja", "imagen": "images/ni3.jpg", "descripcion": "Bruja transformada en mona que roba niños pequeños"}
],
"🇵🇦 Panamá": [
{"nombre": "Humantahú", "imagen": "images/pa1.jpg", "descripcion": "Chamán kuna protector de la naturaleza y animales"},
{"nombre": "La Silampa", "imagen": "images/pa2.jpg", "descripcion": "Mujer fantasma que seduce y mata a hombres solitarios"},
{"nombre": "El Chivato", "imagen": "images/pa3.jpg", "descripcion": "Cabro diabólico que aparece en encrucijadas nocturnas"}
],
"🇵🇾 Paraguay": [
{"nombre": "Mbói Tata", "imagen": "images/py1.png", "descripcion": "Serpiente de fuego protectora de campos y esteros"},
{"nombre": "Urutau", "imagen": "images/py2.jpg", "descripcion": "Ave fantasma cuyo canto anuncia tragedias familiares"},
{"nombre": "Tajy", "imagen": "images/py3.jpg", "descripcion": "Espíritu del lapacho florido en la mitología guaraní"}
],
"🇵🇪 Perú": [
{"nombre": "Wiracocha", "imagen": "images/pe1.jpg", "descripcion": "Dios creador inca señor de todas las cosas vivientes"},
{"nombre": "El Muki", "imagen": "images/pe2.webp", "descripcion": "Duende minero que protege vetas de oro y plata"},
{"nombre": "Los Hermanos Ayar", "imagen": "images/pe3.jpg", "descripcion": "Cuatro hermanos fundadores míticos del imperio inca"}
],
"🇵🇹 Portugal": [
{"nombre": "El Marialva", "imagen": "images/po1.jpg", "descripcion": "Caballero galante y seductor de la tradición portuguesa"},
{"nombre": "El Pez Milagroso", "imagen": "images/po2.webp", "descripcion": "Pez mágico que concede deseos a pescadores devotos"},
{"nombre": "La Cueva da Moeda", "imagen": "images/po3.webp", "descripcion": "Cueva encantada llena de tesoros moros perdidos"}
],
"🇵🇷 Puerto Rico": [
{"nombre": "Yúcahu", "imagen": "images/pr1.jpg", "descripcion": "Dios taíno de la yuca y protector de cosechas"},
{"nombre": "Atabey", "imagen": "images/pr2.jpg", "descripcion": "Diosa madre taína de las aguas dulces y fertilidad"},
{"nombre": "Anacacuya", "imagen": "images/pr3.png", "descripcion": "Estrella taína guía de navegantes y pescadores"}
],
"🇩🇴 República Dominicana": [
{"nombre": "El Galipote", "imagen": "images/rd1.webp", "descripcion": "Brujo que se transforma en animal para hacer maldades"},
{"nombre": "El Lugaru", "imagen": "images/rd2.webp", "descripcion": "Hombre lobo dominicano que ataca ganado y personas"},
{"nombre": "Papa Legba", "imagen": "images/rd3.jpg", "descripcion": "Loa vudú guardián de encrucijadas y comunicaciones"}
],
"🇺🇾 Uruguay": [
{"nombre": "El Yasy Yateré", "imagen": "images/ur1.jpg", "descripcion": "Duende rubio protector de niños con bastón mágico"},
{"nombre": "El Ñandú Barriga Blanca", "imagen": "images/ur2.jpg", "descripcion": "Ñandú gigante y sagrado de las pampas uruguayas"},
{"nombre": "El Pombero", "imagen": "images/ur3.jpg", "descripcion": "Duende travieso protector de aves y la naturaleza"}
],
"🇻🇪 Venezuela": [
{"nombre": "Amalivaca", "imagen": "images/ve1.jpg", "descripcion": "Héroe cultural tamanaco creador del río Orinoco"},
{"nombre": "Osemma", "imagen": "images/ve2.jpg", "descripcion": "Espíritu warao de las palmeras de moriche"},
{"nombre": "La Sayona", "imagen": "images/ve3.webp", "descripcion": "Mujer vengativa que persigue y castiga a infieles"}
]
};
# Variables globales
model = None
tokenizer = None
current_personajes = [] # Para mantener el estado de los personajes actuales
def load_model():
"""Cargar modelo y tokenizador"""
global model, tokenizer
if torch.cuda.is_available():
try:
tokenizer = AutoTokenizer.from_pretrained(MODEL_ID)
model = AutoModelForCausalLM.from_pretrained(
MODEL_ID,
torch_dtype=torch.float32,
device_map="auto",
trust_remote_code=True,
)
if tokenizer.pad_token is None:
tokenizer.pad_token = tokenizer.eos_token
return True
except Exception as e:
print(f"Error GPU: {e}")
return False
else:
try:
local_model_path = os.path.join("models", GGUF_FILENAME)
if os.path.exists(local_model_path):
model_path = local_model_path
else:
model_path = hf_hub_download(
repo_id=GGUF_MODEL_ID,
filename=GGUF_FILENAME,
revision=GGUF_REVISION,
local_dir="./models",
force_download=False,
resume_download=True
)
tokenizer = AutoTokenizer.from_pretrained("google/gemma-3-1b-it")
model = Llama(
model_path=model_path,
n_ctx=2048,
n_threads=4,
n_gpu_layers=0
)
return True
except Exception as e:
print(f"Error GGUF: {e}")
return False
model_loaded = load_model()
def format_chat_history(messages: list, exclude_last_user: bool = True) -> list:
"""Formatea el historial de chat para el modelo"""
formatted_history = []
messages_to_process = messages[:]
if exclude_last_user and messages_to_process and messages_to_process[-1].get("role") == "user":
messages_to_process = messages_to_process[:-1]
for message in messages_to_process:
current_role = message.get("role")
current_content = message.get("content", "").strip()
if current_role == "assistant" and message.get("metadata"):
continue
if not current_content:
continue
if formatted_history and formatted_history[-1]["role"] == current_role:
formatted_history[-1]["content"] += "\n\n" + current_content
else:
formatted_history.append({
"role": current_role,
"content": current_content
})
return formatted_history
def stream_iberotales_response(
user_message: str,
messages: list,
system_message: str = DEFAULT_SYSTEM_MESSAGE,
max_new_tokens: int = DEFAULT_MAX_NEW_TOKENS,
temperature: float = 0.7,
top_p: float = 0.95,
top_k: int = 50,
repetition_penalty: float = 1.2,
) -> Iterator[list]:
"""Genera respuesta con streaming"""
global model, tokenizer
if model is None or tokenizer is None:
messages.append(ChatMessage(role="assistant", content="Error: Modelo no disponible."))
yield messages
return
try:
chat_history = format_chat_history(messages, exclude_last_user=True)
conversation = []
if system_message.strip():
conversation.append({"role": "system", "content": system_message.strip()})
conversation.extend(chat_history)
conversation.append({"role": "user", "content": user_message})
# Validar alternancia
for i in range(1, len(conversation)):
if conversation[i]["role"] == conversation[i-1]["role"] and conversation[i-1]["role"] != "system":
messages.append(ChatMessage(role="assistant", content="Error: Reinicia la conversación."))
yield messages
return
prompt = tokenizer.apply_chat_template(conversation, tokenize=False, add_generation_prompt=True)
response = model(
prompt,
max_tokens=max_new_tokens,
temperature=temperature,
top_p=top_p,
top_k=top_k,
repeat_penalty=repetition_penalty,
stream=True
)
full_response = ""
thinking_message_index = None
solution_message_index = None
in_think_block = False
in_solution_block = False
thinking_complete = False
for chunk in response:
if chunk["choices"][0]["finish_reason"] is None:
new_text = chunk["choices"][0]["text"]
full_response += new_text
# Procesar pensamiento
if "<think>" in full_response and not thinking_complete:
if not in_think_block:
in_think_block = True
if thinking_message_index is None:
messages.append(ChatMessage(
role="assistant",
content="",
metadata={"title": "🤔 Pensando..."}
))
thinking_message_index = len(messages) - 1
think_start = full_response.find("<think>") + 7
if "</think>" in full_response:
think_end = full_response.find("</think>")
current_thinking = full_response[think_start:think_end].strip()
thinking_complete = True
in_think_block = False
else:
current_thinking = full_response[think_start:].strip()
if thinking_message_index is not None:
messages[thinking_message_index] = ChatMessage(
role="assistant",
content=current_thinking,
metadata={"title": "🤔 Pensando..."}
)
yield messages
# Procesar solución
if "<SOLUTION>" in full_response:
if not in_solution_block:
in_solution_block = True
if solution_message_index is None:
messages.append(ChatMessage(role="assistant", content=""))
solution_message_index = len(messages) - 1
solution_start = full_response.find("<SOLUTION>") + 10
if "</SOLUTION>" in full_response:
solution_end = full_response.find("</SOLUTION>")
current_solution = full_response[solution_start:solution_end].strip()
in_solution_block = False
else:
current_solution = full_response[solution_start:].strip()
if solution_message_index is not None and current_solution:
messages[solution_message_index] = ChatMessage(
role="assistant",
content=current_solution
)
yield messages
# Respuesta sin formato
if full_response.strip() and solution_message_index is None:
clean_response = full_response
if "<think>" in clean_response and "</think>" in clean_response:
clean_response = re.sub(r'<think>.*?</think>', '', clean_response, flags=re.DOTALL)
if "<SOLUTION>" in clean_response and "</SOLUTION>" in clean_response:
clean_response = re.sub(r'<SOLUTION>(.*?)</SOLUTION>', r'\1', clean_response, flags=re.DOTALL)
clean_response = clean_response.strip()
if clean_response:
messages.append(ChatMessage(role="assistant", content=clean_response))
yield messages
except Exception as e:
messages.append(ChatMessage(role="assistant", content=f"Error: {str(e)}"))
yield messages
def user_message(msg: str, history: list) -> tuple[str, list]:
"""Añade mensaje del usuario al historial"""
history.append(ChatMessage(role="user", content=msg))
return "", history
def actualizar_personajes(pais_seleccionado):
"""Actualiza la galería de personajes según el país seleccionado"""
global current_personajes
personajes = PERSONAJES_POR_PAIS.get(pais_seleccionado, [])
current_personajes = personajes # Guardamos el estado actual
if not personajes:
return [], "Selecciona un país para ver sus personajes"
# Crear lista de imágenes y etiquetas para la galería
imagenes = []
for p in personajes:
if os.path.exists(p["imagen"]):
imagenes.append((p["imagen"], f"{p['nombre']}: {p['descripcion']}"))
else:
# Imagen placeholder si no existe
imagenes.append(("", f"{p['nombre']}: {p['descripcion']}"))
return imagenes, f"Personajes de {pais_seleccionado}"
def crear_prompt_desde_personaje(evt: gr.SelectData):
"""Crea un prompt basado en el personaje seleccionado"""
global current_personajes
try:
if evt.index is not None and evt.index < len(current_personajes):
personaje = current_personajes[evt.index]
return f"Crea una historia sobre {personaje['nombre']}, {personaje['descripcion']}" #si alguien lee esto, cambiar el dataste a cuenta en lugar de crea
else:
return "Crea una historia sobre un personaje mítico"
except Exception as e:
print(f"Error al crear prompt: {e}")
return "Crea una historia sobre un personaje mítico"
# Aplicar decorador @spaces.GPU si es necesario
if IS_HF_SPACE and SPACES_AVAILABLE and torch.cuda.is_available():
stream_iberotales_response = spaces.GPU(stream_iberotales_response)
# CSS personalizado para mejorar la apariencia
custom_css = """
.gradio-container {
max-width: 1400px !important;
margin: auto;
padding-top: 1.5rem;
}
#galeria .grid-wrap {
max-height: 350px;
overflow-y: auto;
}
#galeria .grid-container {
grid-template-columns: repeat(1, 1fr) !important;
gap: 0.5rem;
}
#galeria .thumbnail-item {
aspect-ratio: 1;
max-height: 100px;
}
#galeria .thumbnail-item img {
object-fit: cover;
width: 100%;
height: 100%;
border-radius: 8px;
}
.header-info {
background: linear-gradient(135deg, #2c3e50 0%, #1a1a2e 100%);
color: white;
padding: 1rem;
border-radius: 12px;
margin-bottom: 1rem;
text-align: center;
}
"""
# Crear la interfaz
with gr.Blocks(fill_height=True, title="Iberotales", css=custom_css) as demo:
# Header con información del proyecto
with gr.Row():
with gr.Column():
gr.HTML("""
<div class="header-info">
<h1>📚 Iberotales</h1>
<p><strong>Autor:</strong> <a href="https://huggingface.co/daqc" target="_blank" style="text-decoration: none;">David Quispe</a> &nbsp;|&nbsp; <a href="https://github.com/mcdaqc/Iberotales" target="_blank" style="text-decoration: none;">GitHub</a> &nbsp;|&nbsp; <a href="https://huggingface.co/somosnlp-hackathon-2025/iberotales-gemma-3-1b-it-es" target="_blank" style="text-decoration: none;">Modelo</a> &nbsp;|&nbsp; <a href="https://huggingface.co/somosnlp-hackathon-2025/iberotales-gemma-3-1b-it-es-finetune-gguf" target="_blank" style="text-decoration: none;">GGUF</a> &nbsp;|&nbsp; <a href="https://huggingface.co/google/gemma-3-1b-it" target="_blank" style="text-decoration: none;">Base</a></p>
<p><em>Generador de historias basadas en mitos y leyendas de Iberoamérica.</em></p>
<p><em>Hackathon SomosNLP 2025</em></p>
</div>
""")
with gr.Row():
# Panel izquierdo - Pokédex de personajes
with gr.Column(scale=1, min_width=320):
gr.Markdown("### 🗃️ Pokédex de Personajes")
pais_dropdown = gr.Dropdown(
choices=list(PERSONAJES_POR_PAIS.keys()),
value="🇦🇷 Argentina",
label="País",
container=False
)
galeria_personajes = gr.Gallery(
value=[],
label="Personajes",
show_label=False,
elem_id="galeria",
columns=1,
rows=4,
height=350,
object_fit="cover",
preview=False # Esto evita que se expanda automáticamente
)
# Panel derecho - Chat
with gr.Column(scale=2):
chatbot = gr.Chatbot(
type="messages",
show_label=False,
height=400
)
with gr.Row():
input_box = gr.Textbox(
placeholder="Escribe tu historia o selecciona un personaje...",
show_label=False,
scale=4,
container=False
)
send_button = gr.Button("📤", scale=1, variant="primary")
with gr.Row():
clear_button = gr.Button("🗑️ Limpiar", scale=1, size="sm")
with gr.Column(scale=3):
with gr.Row():
max_tokens = gr.Slider(100, MAX_MAX_NEW_TOKENS, DEFAULT_MAX_NEW_TOKENS, label="Tokens", container=False)
temperature = gr.Slider(0.1, 2.0, 0.7, label="Temp", container=False)
# Variables de estado
msg_store = gr.State("")
# Eventos
def submit_message(msg, history):
if not msg.strip():
return msg, history
return "", user_message(msg, history)[1]
def generate_response(msg, history, max_tok, temp):
yield from stream_iberotales_response(msg, history, DEFAULT_SYSTEM_MESSAGE, max_tok, temp)
# Actualizar personajes cuando cambia el país
pais_dropdown.change(
fn=actualizar_personajes,
inputs=[pais_dropdown],
outputs=[galeria_personajes, gr.Textbox(visible=False)]
)
# Cargar personajes iniciales
demo.load(
fn=actualizar_personajes,
inputs=[pais_dropdown],
outputs=[galeria_personajes, gr.Textbox(visible=False)]
)
# Crear prompt desde galería
galeria_personajes.select(
fn=crear_prompt_desde_personaje,
outputs=[input_box]
)
# Envío de mensajes
input_box.submit(
lambda msg, hist: (msg, submit_message(msg, hist)[1]),
inputs=[input_box, chatbot],
outputs=[msg_store, chatbot],
queue=False
).then(
generate_response,
inputs=[msg_store, chatbot, max_tokens, temperature],
outputs=chatbot
)
send_button.click(
lambda msg, hist: (msg, submit_message(msg, hist)[1]),
inputs=[input_box, chatbot],
outputs=[msg_store, chatbot],
queue=False
).then(
generate_response,
inputs=[msg_store, chatbot, max_tokens, temperature],
outputs=chatbot
)
clear_button.click(
lambda: ([], "", ""),
outputs=[chatbot, input_box, msg_store],
queue=False
)
# Lanzar aplicación
if __name__ == "__main__":
if model_loaded:
demo.launch(share=False, show_error=True)
else:
print("Error al cargar el modelo.")