import logging
import sys
import os
import re
import base64
import nest_asyncio
import pandas as pd
from pathlib import Path
from typing import Any, Dict, List, Optional
from PIL import Image
import streamlit as st
import torch
from llama_index.core import Settings, SimpleDirectoryReader, StorageContext, Document
from llama_index.core.storage.docstore import SimpleDocumentStore
# from llama_index.llms.ollama import Ollama
# from llama_index.embeddings.ollama import OllamaEmbedding
from llama_index.core.node_parser import LangchainNodeParser
from langchain.text_splitter import RecursiveCharacterTextSplitter
from llama_index.core.storage.chat_store import SimpleChatStore
from llama_index.core.memory import ChatMemoryBuffer
from llama_index.core.query_engine import RetrieverQueryEngine
from llama_index.core.chat_engine import CondensePlusContextChatEngine
from llama_index.retrievers.bm25 import BM25Retriever
from llama_index.core.retrievers import QueryFusionRetriever
from llama_index.vector_stores.chroma import ChromaVectorStore
from llama_index.core import VectorStoreIndex
# from llama_index.llms.huggingface import HuggingFaceLLM
# from llama_index.llms.huggingface_api import HuggingFaceInferenceAPI
# from llama_index.embeddings.huggingface import HuggingFaceEmbedding
import chromadb
#Configuração da imagem da aba
im = Image.open("pngegg.png")
st.set_page_config(page_title = "Chatbot Carômetro", page_icon=im, layout = "wide")
#Removido loop e adicionado os.makedirs
os.makedirs("bm25_retriever", exist_ok=True)
os.makedirs("chat_store", exist_ok=True)
os.makedirs("chroma_db", exist_ok=True)
os.makedirs("documentos", exist_ok=True)
os.makedirs("curadoria", exist_ok=True)
os.makedirs("chroma_db_curadoria", exist_ok=True)
# Configuração do Streamlit
st.sidebar.title("Configuração de LLM")
sidebar_option = st.sidebar.radio("Selecione o LLM", ["gpt-3.5-turbo"])
# logo_url = 'app\logos\logo-sicoob.jpg'
# st.sidebar.image(logo_url)
import base64
#Configuração da imagem da sidebar
with open("sicoob-logo.png", "rb") as f:
data = base64.b64encode(f.read()).decode("utf-8")
st.sidebar.markdown(
f"""
""",
unsafe_allow_html=True,
)
#if sidebar_option == "Ollama":
# Settings.llm = Ollama(model="llama3.2:latest", request_timeout=500.0, num_gpu=1)
# Settings.embed_model = OllamaEmbedding(model_name="nomic-embed-text:latest")
if sidebar_option == "gpt-3.5-turbo":
from llama_index.llms.openai import OpenAI
from llama_index.embeddings.openai import OpenAIEmbedding
Settings.llm = OpenAI(model="gpt-3.5-turbo")
Settings.embed_model = OpenAIEmbedding(model_name="text-embedding-ada-002")
# elif sidebar_option == 'NuExtract-1.5':
# #Embedding do huggingface
# Settings.embed_model = HuggingFaceEmbedding(
# model_name="BAAI/bge-small-en-v1.5"
# )
# #Carregamento do modelo local, descomentar o modelo desejado
# llm = HuggingFaceLLM(
# context_window=2048,
# max_new_tokens=2048,
# generate_kwargs={"do_sample": False},
# #query_wrapper_prompt=query_wrapper_prompt,
# #model_name="Qwen/Qwen2.5-Coder-32B-Instruct",
# #model_name="Qwen/Qwen2.5-14B-Instruct",
# # model_name="meta-llama/Llama-3.2-3B",
# #model_name="HuggingFaceH4/zephyr-7b-beta",
# # model_name="meta-llama/Meta-Llama-3-8B",
# model_name="numind/NuExtract-1.5",
# #model_name="meta-llama/Llama-3.2-3B",
# tokenizer_name="numind/NuExtract-1.5",
# device_map="auto",
# tokenizer_kwargs={"max_length": 512},
# # uncomment this if using CUDA to reduce memory usage
# model_kwargs={"torch_dtype": torch.bfloat16},
# )
# chat = [
# {"role": "user", "content": "Hello, how are you?"},
# {"role": "assistant", "content": "I'm doing great. How can I help you today?"},
# {"role": "user", "content": "I'd like to show off how chat templating works!"},
# ]
# from transformers import AutoTokenizer
# tokenizer = AutoTokenizer.from_pretrained("numind/NuExtract-1.5")
# tokenizer.apply_chat_template(chat, tokenize=False)
# Settings.chunk_size = 512
# Settings.llm = llm
else:
raise Exception("Opção de LLM inválida!")
logging.basicConfig(stream=sys.stdout, level=logging.INFO)
logging.getLogger().addHandler(logging.StreamHandler(stream=sys.stdout))
# Diretórios configurados pelo usuário
chat_store_path = os.path.join("chat_store", "chat_store.json")
documents_path = os.path.join("documentos")
chroma_storage_path = os.path.join("chroma_db") # Diretório para persistência do Chroma
chroma_storage_path_curadoria = os.path.join("chroma_db_curadoria") # Diretório para 'curadoria'
bm25_persist_path = os.path.join("bm25_retriever")
curadoria_path = os.path.join("curadoria")
# Classe CSV Customizada (novo código)
class CustomPandasCSVReader:
"""PandasCSVReader modificado para incluir cabeçalhos nos documentos."""
def __init__(
self,
*args: Any,
concat_rows: bool = True,
col_joiner: str = ", ",
row_joiner: str = "\n",
pandas_config: dict = {},
**kwargs: Any
) -> None:
self._concat_rows = concat_rows
self._col_joiner = col_joiner
self._row_joiner = row_joiner
self._pandas_config = pandas_config
def load_data(
self,
file: Path,
extra_info: Optional[Dict] = None,
) -> List[Document]:
df = pd.read_csv(file, **self._pandas_config)
text_list = [" ".join(df.columns.astype(str))]
text_list += (
df.astype(str)
.apply(lambda row: self._col_joiner.join(row.values), axis=1)
.tolist()
)
metadata = {"filename": file.name, "extension": file.suffix}
if extra_info:
metadata.update(extra_info)
if self._concat_rows:
return [Document(text=self._row_joiner.join(text_list), metadata=metadata)]
else:
return [
Document(text=text, metadata=metadata)
for text in text_list
]
def clean_documents(documents):
"""Remove caracteres não desejados diretamente nos textos dos documentos."""
cleaned_documents = []
for doc in documents:
cleaned_text = re.sub(r"[^0-9A-Za-zÀ-ÿ ]", "", doc.get_content())
doc.text = cleaned_text
cleaned_documents.append(doc)
return cleaned_documents
from llama_index.readers.google import GoogleDriveReader
import json
credentials_json = os.getenv('GOOGLE_CREDENTIALS')
token_json = os.getenv('GOOGLE_TOKEN')
if credentials_json is None:
raise ValueError("The GOOGLE_CREDENTIALS environment variable is not set.")
# Write the credentials to a file
credentials_path = "credentials.json"
token_path = "token.json"
with open(credentials_path, 'w') as credentials_file:
credentials_file.write(credentials_json)
with open(token_path, 'w') as credentials_file:
credentials_file.write(token_json)
google_drive_reader = GoogleDriveReader(credentials_path=credentials_path)
google_drive_reader._creds = google_drive_reader._get_credentials()
def are_docs_downloaded(directory_path: str) -> bool:
return os.path.isdir(directory_path) and any(os.scandir(directory_path))
def download_original_files_from_folder(greader: GoogleDriveReader, pasta_documentos_drive: str, local_path: str):
os.makedirs(local_path, exist_ok=True)
files_meta = greader._get_fileids_meta(folder_id=pasta_documentos_drive)
if not files_meta:
logging.info("Nenhum arquivo encontrado na pasta especificada.")
return
for fmeta in files_meta:
file_id = fmeta[0]
file_name = os.path.basename(fmeta[2])
local_file_path = os.path.join(local_path, file_name)
if os.path.exists(local_file_path):
logging.info(f"Arquivo '{file_name}' já existe localmente, ignorando download.")
continue
downloaded_file_path = greader._download_file(file_id, local_file_path)
if downloaded_file_path:
logging.info(f"Arquivo '{file_name}' baixado com sucesso em: {downloaded_file_path}")
else:
logging.warning(f"Não foi possível baixar '{file_name}'")
#DADOS/QA_database/Documentos CSV/documentos
pasta_documentos_drive = "1xVzo8s1D0blzR5ZB3m5k4dVWHuRmKUu-"
#DADOS/QA_database/Documentos CSV/curadoria
pasta_curadoria_drive = "1LRrdOkZy9p0FA3MQAyz-Ssj3ktKTWAwE"
# Verifica e baixa arquivos se necessário (novo código)
if not are_docs_downloaded(documents_path):
logging.info("Baixando arquivos originais do Drive para 'documentos'...")
download_original_files_from_folder(google_drive_reader, pasta_documentos_drive, documents_path)
else:
logging.info("'documentos' já contém arquivos, ignorando download.")
if not are_docs_downloaded(curadoria_path):
logging.info("Baixando arquivos originais do Drive para 'curadoria'...")
download_original_files_from_folder(google_drive_reader, pasta_curadoria_drive, curadoria_path)
else:
logging.info("'curadoria' já contém arquivos, ignorando download.")
# Configuração de leitura de documentos
file_extractor = {".csv": CustomPandasCSVReader()}
documents = SimpleDirectoryReader(
input_dir=documents_path,
file_extractor=file_extractor,
filename_as_id=True
).load_data()
documents = clean_documents(documents)
# Configuração do Chroma e BM25 com persistência
docstore = SimpleDocumentStore()
docstore.add_documents(documents)
db = chromadb.PersistentClient(path=chroma_storage_path)
chroma_collection = db.get_or_create_collection("dense_vectors")
vector_store = ChromaVectorStore(chroma_collection=chroma_collection)
# Configuração do StorageContext
storage_context = StorageContext.from_defaults(
docstore=docstore, vector_store=vector_store
)
# Criação/Recarregamento do índice com embeddings
if os.path.exists(chroma_storage_path):
index = VectorStoreIndex.from_vector_store(vector_store)
else:
splitter = LangchainNodeParser(
RecursiveCharacterTextSplitter(chunk_size=512, chunk_overlap=64)
)
index = VectorStoreIndex.from_documents(
documents,
storage_context=storage_context,
transformations=[splitter]
)
vector_store.persist()
# Criação/Recarregamento do BM25 Retriever
if os.path.exists(os.path.join(bm25_persist_path, "params.index.json")):
bm25_retriever = BM25Retriever.from_persist_dir(bm25_persist_path)
else:
bm25_retriever = BM25Retriever.from_defaults(
docstore=docstore,
similarity_top_k=2,
language="portuguese", # Idioma ajustado para seu caso
)
os.makedirs(bm25_persist_path, exist_ok=True)
bm25_retriever.persist(bm25_persist_path)
#Adicionado documentos na pasta curadoria, foi setado para 1200 o chunk pra receber pergunta, contexto e resposta
curadoria_documents = SimpleDirectoryReader(
input_dir=curadoria_path,
file_extractor=file_extractor,
filename_as_id=True
).load_data()
curadoria_documents = clean_documents(curadoria_documents)
curadoria_docstore = SimpleDocumentStore()
curadoria_docstore.add_documents(curadoria_documents)
db_curadoria = chromadb.PersistentClient(path=chroma_storage_path_curadoria)
chroma_collection_curadoria = db_curadoria.get_or_create_collection("dense_vectors_curadoria")
vector_store_curadoria = ChromaVectorStore(chroma_collection=chroma_collection_curadoria)
# Configuração do StorageContext para 'curadoria'
storage_context_curadoria = StorageContext.from_defaults(
docstore=curadoria_docstore, vector_store=vector_store_curadoria
)
# Criação/Recarregamento do índice com embeddings para 'curadoria'
if os.path.exists(chroma_storage_path_curadoria):
curadoria_index = VectorStoreIndex.from_vector_store(vector_store_curadoria)
else:
curadoria_splitter = LangchainNodeParser(
RecursiveCharacterTextSplitter(chunk_size=1200, chunk_overlap=100)
)
curadoria_index = VectorStoreIndex.from_documents(
curadoria_documents, storage_context=storage_context_curadoria, transformations=[curadoria_splitter]
)
vector_store_curadoria.persist()
curadoria_retriever = curadoria_index.as_retriever(similarity_top_k=2)
# Combinação de Retrievers (Embeddings + BM25)
vector_retriever = index.as_retriever(similarity_top_k=2)
retriever = QueryFusionRetriever(
[vector_retriever, bm25_retriever, curadoria_retriever],
similarity_top_k=2,
num_queries=0,
mode="reciprocal_rerank",
use_async=True,
verbose=True,
query_gen_prompt=(
"Gere {num_queries} perguntas de busca relacionadas à seguinte pergunta. "
"Priorize o significado da pergunta sobre qualquer histórico de conversa. "
"Se o histórico não for relevante para a pergunta, ignore-o. "
"Não adicione explicações, notas ou introduções. Apenas escreva as perguntas. "
"Pergunta: {query}\n\n"
"Perguntas:\n"
),
)
# Configuração do chat engine
nest_asyncio.apply()
memory = ChatMemoryBuffer.from_defaults(token_limit=3900)
query_engine = RetrieverQueryEngine.from_args(retriever)
chat_engine = CondensePlusContextChatEngine.from_defaults(
query_engine,
memory=memory,
context_prompt=(
"Você é um assistente virtual capaz de interagir normalmente, além de"
" fornecer informações sobre organogramas e listar funcionários."
" Aqui estão os documentos relevantes para o contexto:\n"
"{context_str}"
"\nInstrução: Use o histórico da conversa anterior, ou o contexto acima, para responder."
),
verbose=True,
)
# Armazenamento do chat
chat_store = SimpleChatStore()
if os.path.exists(chat_store_path):
chat_store = SimpleChatStore.from_persist_path(persist_path=chat_store_path)
else:
chat_store.persist(persist_path=chat_store_path)
# Interface do Chatbot
st.title("Chatbot Carômetro")
st.write("Este chatbot pode te ajudar a conseguir informações relevantes sobre os carômetros da Sicoob.")
if 'chat_history' not in st.session_state:
st.session_state.chat_history = []
for message in st.session_state.chat_history:
role, text = message.split(":", 1)
with st.chat_message(role.strip().lower()):
st.write(text.strip())
user_input = st.chat_input("Digite sua pergunta")
if user_input:
# Exibir a mensagem do usuário e adicionar ao histórico
with st.chat_message('user'):
st.write(user_input)
st.session_state.chat_history.append(f"user: {user_input}")
# Placeholder para a mensagem do assistente
with st.chat_message('assistant'):
message_placeholder = st.empty()
assistant_message = ''
# Obter a resposta em streaming do chat_engine
response = chat_engine.stream_chat(user_input)
for token in response.response_gen:
assistant_message += token
# Atualizar o placeholder da mensagem
message_placeholder.markdown(assistant_message + "▌")
# Remover o cursor após a conclusão
message_placeholder.markdown(assistant_message)
st.session_state.chat_history.append(f"assistant: {assistant_message}")