File size: 5,832 Bytes
cb92864 08fb91a cb92864 08fb91a cb92864 76d8844 cb92864 7f57ffc cb92864 7ec56d7 cb92864 7ec56d7 cb92864 7ec56d7 | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 | """
app.py β FinNode GraphRAG μ±λ΄
================================
Hugging Face Spaces λ°°ν¬ μ§μ
μ .
Gradio ChatInterface + LangGraph κΈ°λ° λν νλ¦ μ μ΄.
μ€ν:
python app.py
"""
from typing import List, TypedDict
import dotenv
import gradio as gr
from langgraph.graph import END, StateGraph
from src.retrieval.finRetrieval import graphrag
dotenv.load_dotenv()
# ββββββββββββββββββββββββββββββββββββββββββ
# Startup DB μκ° μ§λ¨ (Fail-Fast)
# ββββββββββββββββββββββββββββββββββββββββββ
# νκΉ
νμ΄μ€Spaces λ° μ€μ μ± μλ² κ΅¬λ μμ μλ μ¦μ μκ° μ§λ¨μ μννμ¬,
# Neo4j λ°μ΄ν°λ² μ΄μ€ μ°κ²°μ΄ λΆκ°λ₯νλ©΄ ꡬλ μ€ν¨(Crash Early)λ₯Ό μΌμΌν΅λλ€.
try:
graphrag._init_once()
print("β
[μκ° μ§λ¨ μλ£] Neo4j AuraDB μ§μ κ·Έλνμ μλ²½νκ² μ μλμμ΅λλ€!")
except Exception as e:
print(f"β [μκ° μ§λ¨ μ€ν¨] Neo4j DB μ°κ²° νμΈ μ€ μλ¬κ° λ°μνμ΅λλ€: {e}")
raise e
# ββββββββββββββββββββββββββββββββββββββββββ
# 1. LangGraph μ±λ΄ State μ μ
# ββββββββββββββββββββββββββββββββββββββββββ
class ChatState(TypedDict):
question: str # μ¬μ©μ μ§λ¬Έ
history: List[dict] # λν νμ€ν 리 [{"role": "user"/"assistant", "content": "..."}]
context: str # GraphRAG κ²μ κ²°κ³Ό
answer: str # μ΅μ’
λ΅λ³
# ββββββββββββββββββββββββββββββββββββββββββ
# 2. LangGraph λ
Έλ μ μ
# ββββββββββββββββββββββββββββββββββββββββββ
def retrieve_node(state: ChatState) -> ChatState:
"""Node 1: GraphRAGλ‘ κ΄λ ¨ 컨ν
μ€νΈ κ²μ"""
try:
result = graphrag.search(query_text=state["question"])
context = result.answer # GraphRAGκ° μ΄λ―Έ λ΅λ³μ μμ±νλ―λ‘ λ°λ‘ μ¬μ©
except Exception as e:
context = f"[κ²μ μ€λ₯: {e}]"
return {**state, "context": context}
def generate_node(state: ChatState) -> ChatState:
"""Node 2: λν νμ€ν 리λ₯Ό κ³ λ €νμ¬ μ΅μ’
λ΅λ³ μμ±
GraphRAGκ° μ΄λ―Έ κ²μ + μμ±μ μ²λ¦¬νλ―λ‘,
μ¬κΈ°μλ νμ€ν 리 κΈ°λ° νμ²λ¦¬λ μΆκ° ν¬λ§·ν
λ§ μνν©λλ€.
"""
# GraphRAG κ²°κ³Όλ₯Ό λ°λ‘ λ΅λ³μΌλ‘ μ¬μ©
# (νμ€ν 리 κΈ°λ° νμ μ§λ¬Έ μ²λ¦¬κ° νμνλ©΄ μ΄ λ
Έλλ₯Ό νμ₯νμΈμ)
answer = state["context"] if state["context"] else "κ΄λ ¨ μ 보λ₯Ό μ°Ύμ μ μμ΅λλ€."
return {**state, "answer": answer}
# ββββββββββββββββββββββββββββββββββββββββββ
# 3. LangGraph μν¬νλ‘μ° μ»΄νμΌ
# ββββββββββββββββββββββββββββββββββββββββββ
builder = StateGraph(ChatState)
builder.add_node("retrieve", retrieve_node)
builder.add_node("generate", generate_node)
builder.set_entry_point("retrieve")
builder.add_edge("retrieve", "generate")
builder.add_edge("generate", END)
chat_graph = builder.compile()
# ββββββββββββββββββββββββββββββββββββββββββ
# 4. Gradio μ°λ ν¨μ
# ββββββββββββββββββββββββββββββββββββββββββ
def chat(message: str, history: list) -> str:
"""Gradio ChatInterfaceκ° νΈμΆνλ ν¨μ.
Args:
message: μ¬μ©μ μ
λ ₯ λ©μμ§
history: Gradioκ° κ΄λ¦¬νλ λν νμ€ν 리
[{"role": "user"/"assistant", "content": "..."}] νμ
Returns:
str: μ±λ΄ λ΅λ³
"""
if not message.strip():
return "μ§λ¬Έμ μ
λ ₯ν΄ μ£ΌμΈμ."
# Gradio history β LangGraph state νμμΌλ‘ λ³ν
state: ChatState = {
"question": message,
"history": history,
"context": "",
"answer": "",
}
result = chat_graph.invoke(state)
return result["answer"]
# ββββββββββββββββββββββββββββββββββββββββββ
# 5. Gradio UI ꡬμ±
# ββββββββββββββββββββββββββββββββββββββββββ
demo = gr.ChatInterface(
fn=chat,
chatbot=gr.Chatbot(height=500),
textbox=gr.Textbox(container=False, scale=7),
title="FinNode β AI κΈ°μ
νΈλ λ λΆμ μ±λ΄",
description=(
"> μ΅μ AI λ΄μ€λ₯Ό κΈ°λ°μΌλ‘ ꡬμΆλ μ§μ κ·Έλν(GraphRAG)μμ λ΅λ³ν©λλ€.\n\n"
"**μμ μ§λ¬Έ**\n"
"- μΌμ±μ μμ μ΅κ·Ό AI κΈ°μ νΈλ λλ?\n"
"- μΉ΄μΉ΄μ€κ° κ°λ° μ€μΈ AI μλΉμ€ λͺ©λ‘μ μλ €μ€\n"
"- μ΄λ€ κΈ°μ
μ΄ LLM κΈ°μ μ κ°λ°νλμ?\n"
"- μ΅κ·Ό AI κ΄λ ¨ λ΄μ€ κΈ°μ¬λ₯Ό μμ½ν΄μ€"
),
examples=[
"μΌμ±μ μμ μ΅κ·Ό AI κΈ°μ νΈλ λλ?",
"μΉ΄μΉ΄μ€κ° κ°λ° μ€μΈ AI μλΉμ€ λͺ©λ‘μ μλ €μ€",
"μ΄λ€ κΈ°μ
μ΄ LLM κΈ°μ μ κ°λ°νλμ?",
"μ΅κ·Ό AI κ΄λ ¨ λ΄μ€ κΈ°μ¬λ₯Ό μμ½ν΄μ€",
],
cache_examples=False,
)
if __name__ == "__main__":
demo.launch(
server_name="0.0.0.0",
server_port=7860,
theme=gr.themes.Soft(primary_hue="indigo")
)
|