AgentX-Papers / app.py
openfree's picture
Update app.py
71b8a5a verified
import streamlit as st
import os
import json
from datetime import datetime, timedelta
import base64
import pandas as pd
import pydeck as pdk
from paper import (
literature_research_task, outline_task, draft_writing_task,
citation_task, editing_task, chatbot_task,
run_task
)
# st.set_page_config()๋Š” ๋‹ค๋ฅธ Streamlit ํ•จ์ˆ˜๋ณด๋‹ค ๊ฐ€์žฅ ๋จผ์ € ์‹คํ–‰๋˜์–ด์•ผ ํ•ฉ๋‹ˆ๋‹ค.
st.set_page_config(
page_title="Your AI Agent for Academic Research",
page_icon="๐Ÿ“š",
layout="wide",
initial_sidebar_state="expanded"
)
# ------------------------------------------
# ๋‹ค๊ตญ์–ด ์ง€์› (์˜์–ด/ํ•œ๊ตญ์–ด ์˜ˆ์‹œ)
# ------------------------------------------
translations = {
"en": {
"page_title": "Your AI Agent for Academic Research",
"header": "Your AI Agent for Academic Research",
"create_itinerary": "Generate Your Research Paper",
"trip_details": "Research Details",
"origin": "Research Topic",
"destination": "Paper Title",
"travel_dates": "Due Date",
"duration": "Paper Length (pages)",
"preferences": "Keywords/Focus",
"special_requirements": "Additional Instructions",
"submit": "๐Ÿš€ Generate My Research Paper",
"request_details": "Your Research Request",
"from": "Topic",
"when": "Due Date",
"budget": "Paper Type",
"travel_style": "Writing Style",
"live_agent_outputs": "Live Agent Outputs",
"full_itinerary": "Full Paper",
"details": "Details",
"download_share": "Download & Share",
"save_itinerary": "Save Your Paper",
"plan_another_trip": "๐Ÿ”„ Generate Another Paper",
"about": "About",
"how_it_works": "How it works",
"travel_agents": "Research Agents",
"share_itinerary": "Share Your Paper",
"save_for_mobile": "Save for Mobile",
"built_with": "Built with โค๏ธ for you",
"itinerary_ready": "Your Research Paper is Ready! ๐ŸŽ‰",
"personalized_experience": "We've created a personalized academic paper based on your inputs. Explore your paper below.",
"agent_activity": "Agent Activity",
"error_origin_destination": "Please enter both the research topic and paper title.",
"your_itinerary_file": "Your Paper File",
"text_format": "Text format - Can be opened in any text editor"
},
"ko": {
"page_title": "๋‹น์‹ ์˜ ํ•™์ˆ  ์—ฐ๊ตฌ AI ์—์ด์ „ํŠธ",
"header": "๋‹น์‹ ์˜ ํ•™์ˆ  ์—ฐ๊ตฌ AI ์—์ด์ „ํŠธ",
"create_itinerary": "๋…ผ๋ฌธ ์ƒ์„ฑ",
"trip_details": "์—ฐ๊ตฌ ์„ธ๋ถ€์‚ฌํ•ญ",
"origin": "์—ฐ๊ตฌ ์ฃผ์ œ",
"destination": "๋…ผ๋ฌธ ์ œ๋ชฉ",
"travel_dates": "์ œ์ถœ ๊ธฐํ•œ",
"duration": "๋…ผ๋ฌธ ๋ถ„๋Ÿ‰ (ํŽ˜์ด์ง€)",
"preferences": "ํ‚ค์›Œ๋“œ/์ฃผ์š” ์ดˆ์ ",
"special_requirements": "์ถ”๊ฐ€ ์ง€์‹œ์‚ฌํ•ญ",
"submit": "๐Ÿš€ ๋‚˜์˜ ๋…ผ๋ฌธ ์ƒ์„ฑ",
"request_details": "์—ฐ๊ตฌ ์š”์ฒญ ์ •๋ณด",
"from": "์ฃผ์ œ",
"when": "์ œ์ถœ ๊ธฐํ•œ",
"budget": "๋…ผ๋ฌธ ์ข…๋ฅ˜",
"travel_style": "์ž‘์„ฑ ์Šคํƒ€์ผ",
"live_agent_outputs": "์‹ค์‹œ๊ฐ„ ์—์ด์ „ํŠธ ๊ฒฐ๊ณผ",
"full_itinerary": "์ „์ฒด ๋…ผ๋ฌธ",
"details": "์„ธ๋ถ€์‚ฌํ•ญ",
"download_share": "๋‹ค์šด๋กœ๋“œ ๋ฐ ๊ณต์œ ",
"save_itinerary": "๋…ผ๋ฌธ ์ €์žฅ",
"plan_another_trip": "๐Ÿ”„ ๋‹ค๋ฅธ ๋…ผ๋ฌธ ์ƒ์„ฑ",
"about": "์†Œ๊ฐœ",
"how_it_works": "์ž‘๋™ ๋ฐฉ์‹",
"travel_agents": "์—ฐ๊ตฌ ์—์ด์ „ํŠธ",
"share_itinerary": "๋…ผ๋ฌธ ๊ณต์œ ",
"save_for_mobile": "๋ชจ๋ฐ”์ผ ์ €์žฅ",
"built_with": "๋‹น์‹ ์„ ์œ„ํ•ด โค๏ธ ๋งŒ๋“ค์–ด์กŒ์Šต๋‹ˆ๋‹ค",
"itinerary_ready": "๋…ผ๋ฌธ์ด ์ค€๋น„๋˜์—ˆ์Šต๋‹ˆ๋‹ค! ๐ŸŽ‰",
"personalized_experience": "์ž…๋ ฅํ•˜์‹  ์ •๋ณด๋ฅผ ๋ฐ”ํƒ•์œผ๋กœ ๋งž์ถคํ˜• ๋…ผ๋ฌธ์ด ์ƒ์„ฑ๋˜์—ˆ์Šต๋‹ˆ๋‹ค. ์•„๋ž˜์—์„œ ๋…ผ๋ฌธ์„ ํ™•์ธํ•˜์„ธ์š”.",
"agent_activity": "์—์ด์ „ํŠธ ํ™œ๋™",
"error_origin_destination": "์—ฐ๊ตฌ ์ฃผ์ œ์™€ ๋…ผ๋ฌธ ์ œ๋ชฉ์„ ๋ชจ๋‘ ์ž…๋ ฅํ•˜์„ธ์š”.",
"your_itinerary_file": "๋…ผ๋ฌธ ํŒŒ์ผ",
"text_format": "ํ…์ŠคํŠธ ํ˜•์‹ - ๋ชจ๋“  ํ…์ŠคํŠธ ํŽธ์ง‘๊ธฐ์—์„œ ์—ด ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค."
}
}
def t(key):
lang = st.session_state.get("selected_language", "en")
return translations[lang].get(key, key)
# ---------------------------
# ์„ธ์…˜ ์ดˆ๊ธฐํ™”
# ---------------------------
if 'selected_language' not in st.session_state:
st.session_state.selected_language = "en"
# ------------------------------------------
# ์‚ฌ์ด๋“œ๋ฐ”์— ์–ธ์–ด ์„ ํƒ ์œ„์ ฏ ์ถ”๊ฐ€
# ------------------------------------------
with st.sidebar:
language = st.selectbox(
"Language / ์–ธ์–ด",
["English", "ํ•œ๊ตญ์–ด"]
)
lang_map = {
"English": "en",
"ํ•œ๊ตญ์–ด": "ko"
}
st.session_state.selected_language = lang_map.get(language, "en")
# ------------------------------------------
# UI ์‹œ์ž‘
# ------------------------------------------
st.markdown("""
<style>
:root {
--primary: #3a86ff;
--primary-light: #4895ef;
--primary-dark: #2667ff;
--background: #f8f9fa;
--card-bg: #ffffff;
--text: #212529;
--border: #e9ecef;
}
.main-header {
font-size: 2.5rem;
color: var(--primary-dark);
text-align: center;
margin-bottom: 0.8rem;
font-weight: 700;
}
.modern-card {
background-color: var(--card-bg);
border-radius: 10px;
padding: 1.2rem;
margin-bottom: 1.2rem;
box-shadow: 0 2px 10px rgba(0,0,0,0.05);
border: 1px solid var(--border);
}
</style>
""", unsafe_allow_html=True)
def get_download_link(text_content, filename):
b64 = base64.b64encode(text_content.encode()).decode()
href = f'<a class="download-link" href="data:text/plain;base64,{b64}" download="{filename}"><i>๐Ÿ“ฅ</i> {t("save_itinerary")}</a>'
return href
def display_modern_progress(current_step, total_steps=5):
if 'progress_steps' not in st.session_state:
st.session_state.progress_steps = {
0: {'status': 'pending', 'name': t("trip_details")},
1: {'status': 'pending', 'name': t("about")},
2: {'status': 'pending', 'name': t("live_agent_outputs")},
3: {'status': 'pending', 'name': t("download_share")},
4: {'status': 'pending', 'name': t("full_itinerary")}
}
for i in range(total_steps):
if i < current_step:
st.session_state.progress_steps[i]['status'] = 'complete'
elif i == current_step:
st.session_state.progress_steps[i]['status'] = 'active'
else:
st.session_state.progress_steps[i]['status'] = 'pending'
progress_percentage = (current_step / total_steps) * 100
st.progress(progress_percentage / 100)
st.markdown("<div>Progress: " + str(progress_percentage) + "% completed.</div>")
return progress_percentage
def update_step_status(step_index, status):
if 'progress_steps' in st.session_state and step_index in st.session_state.progress_steps:
st.session_state.progress_steps[step_index]['status'] = status
def run_task_with_logs(task, input_text, log_container, output_container, results_key=None):
log_message = f"๐Ÿค– Starting {task.agent.role}..."
st.session_state.log_messages.append(log_message)
with log_container:
st.markdown("### " + t("agent_activity"))
for msg in st.session_state.log_messages:
st.markdown(msg)
result = run_task(task, input_text)
if results_key:
st.session_state.results[results_key] = result
log_message = f"โœ… {task.agent.role} completed!"
st.session_state.log_messages.append(log_message)
with log_container:
st.markdown("### " + t("agent_activity"))
for msg in st.session_state.log_messages:
st.markdown(msg)
with output_container:
st.markdown(f"### {task.agent.role} Output")
st.markdown("<div class='agent-output'>" + result + "</div>", unsafe_allow_html=True)
return result
if 'generated_itinerary' not in st.session_state:
st.session_state.generated_itinerary = None
if 'generation_complete' not in st.session_state:
st.session_state.generation_complete = False
if 'current_step' not in st.session_state:
st.session_state.current_step = 0
if 'results' not in st.session_state:
st.session_state.results = {
"literature_review": "",
"outline": "",
"draft": "",
"citations": "",
"edited": ""
}
if 'log_messages' not in st.session_state:
st.session_state.log_messages = []
if 'form_submitted' not in st.session_state:
st.session_state.form_submitted = False
st.markdown(f"""
<div style="text-align: center;">
<img src="https://img.icons8.com/fluency/96/book.png" width="90">
<h1 class="main-header">{t("header")}</h1>
<p>Generate your personalized research paper with AI-powered academic agents.</p>
</div>
""", unsafe_allow_html=True)
st.markdown('<hr>', unsafe_allow_html=True)
with st.sidebar:
st.markdown("""
<div style="text-align: center;">
<img src="https://img.icons8.com/fluency/96/book.png" width="80">
<h3>Your AI Academic Research Assistant</h3>
<p>AI-Powered Paper Generation</p>
</div>
""", unsafe_allow_html=True)
st.markdown('<div class="modern-card">', unsafe_allow_html=True)
st.markdown("### " + t("about"))
st.info("This tool generates a personalized academic research paper based on your inputs. Fill in the form and let our specialized agents craft your paper!")
st.markdown('</div>', unsafe_allow_html=True)
st.markdown('<div class="modern-card">', unsafe_allow_html=True)
st.markdown("### " + t("how_it_works"))
st.markdown("""
<ol>
<li>Enter your research details</li>
<li>AI conducts literature research</li>
<li>Generate a paper outline</li>
<li>Draft and edit your paper</li>
<li>Download your final paper</li>
</ol>
""", unsafe_allow_html=True)
st.markdown('</div>', unsafe_allow_html=True)
if not st.session_state.generation_complete:
st.markdown('<div class="modern-card">', unsafe_allow_html=True)
st.markdown("<h3>" + t("create_itinerary") + "</h3>", unsafe_allow_html=True)
st.markdown("<p>Fill in the details below to generate your research paper.</p>", unsafe_allow_html=True)
with st.form("research_form"):
col1, col2 = st.columns(2)
with col1:
research_topic = st.text_input(t("origin"), placeholder="e.g., Deep Learning in Healthcare")
paper_title = st.text_input(t("destination"), placeholder="e.g., Advances in Deep Learning for Medical Diagnosis")
due_date = st.date_input(t("travel_dates"), min_value=datetime.now())
with col2:
paper_length = st.slider(t("duration"), min_value=5, max_value=50, value=10)
paper_type_options = ["Journal", "Conference", "Thesis", "Review"]
paper_type = st.selectbox(t("budget"), paper_type_options, help="Select the type of paper")
writing_style = st.multiselect(t("travel_style"), options=["Formal", "Technical", "Creative"], default=["Formal"])
additional_instructions = st.text_area(t("special_requirements"), placeholder="Any additional instructions or requirements...")
keywords = st.text_area(t("preferences"), placeholder="Enter keywords or focus areas, separated by commas")
submit_button = st.form_submit_button(t("submit"))
st.markdown('</div>', unsafe_allow_html=True)
if submit_button:
if not research_topic or not paper_title:
st.error(t("error_origin_destination"))
else:
st.session_state.form_submitted = True
st.session_state.research_topic = research_topic
user_input = {
"research_topic": research_topic,
"paper_title": paper_title,
"due_date": due_date.strftime("%Y-%m-%d"),
"paper_length": str(paper_length),
"paper_type": paper_type,
"writing_style": ", ".join(writing_style),
"keywords": keywords,
"additional_instructions": additional_instructions
}
st.session_state.user_input = user_input
input_context = f"""Research Request Details:
Research Topic: {user_input['research_topic']}
Paper Title: {user_input['paper_title']}
Due Date: {user_input['due_date']}
Paper Length: {user_input['paper_length']} pages
Paper Type: {user_input['paper_type']}
Writing Style: {user_input['writing_style']}
Keywords/Focus: {user_input['keywords']}
Additional Instructions: {user_input['additional_instructions']}
"""
llm_language_instructions = {
"en": "Please output the response in English.",
"ko": "ํ•œ๊ตญ์–ด๋กœ ์ถœ๋ ฅํ•ด ์ฃผ์„ธ์š”."
}
selected_lang = st.session_state.get("selected_language", "en")
language_instruction = llm_language_instructions.get(selected_lang, "Please output the response in English.")
modified_input_context = language_instruction + "\n" + input_context
st.markdown("<div>Processing your request...</div>", unsafe_allow_html=True)
st.session_state.current_step = 0
update_step_status(0, 'active')
progress_placeholder = st.empty()
with progress_placeholder.container():
display_modern_progress(st.session_state.current_step)
log_container = st.container()
st.session_state.log_messages = []
output_container = st.container()
st.session_state.results = {}
# Step 1: Literature Research
literature_review = run_task_with_logs(
literature_research_task,
modified_input_context.format(topic=user_input['research_topic'], keywords=user_input['keywords']),
log_container,
output_container,
"literature_review"
)
update_step_status(0, 'complete')
st.session_state.current_step = 1
update_step_status(1, 'active')
with progress_placeholder.container():
display_modern_progress(st.session_state.current_step)
# Step 2: Generate Outline
outline = run_task_with_logs(
outline_task,
modified_input_context.format(topic=user_input['research_topic']),
log_container,
output_container,
"outline"
)
update_step_status(1, 'complete')
st.session_state.current_step = 2
update_step_status(2, 'active')
with progress_placeholder.container():
display_modern_progress(st.session_state.current_step)
# Step 3: Draft Writing
draft = run_task_with_logs(
draft_writing_task,
modified_input_context.format(topic=user_input['research_topic']),
log_container,
output_container,
"draft"
)
update_step_status(2, 'complete')
st.session_state.current_step = 3
update_step_status(3, 'active')
with progress_placeholder.container():
display_modern_progress(st.session_state.current_step)
# Step 4: Citation Generation
citations = run_task_with_logs(
citation_task,
modified_input_context.format(topic=user_input['research_topic']),
log_container,
output_container,
"citations"
)
update_step_status(3, 'complete')
st.session_state.current_step = 4
update_step_status(4, 'active')
with progress_placeholder.container():
display_modern_progress(st.session_state.current_step)
# Step 5: Editing and Polishing
edited = run_task_with_logs(
editing_task,
modified_input_context.format(topic=user_input['research_topic']),
log_container,
output_container,
"edited"
)
update_step_status(4, 'complete')
st.session_state.current_step = 5
with progress_placeholder.container():
display_modern_progress(st.session_state.current_step)
full_paper = f"""Research Paper:
{input_context}
Literature Review:
{literature_review}
Outline:
{outline}
Draft:
{draft}
Citations:
{citations}
Edited Version:
{edited}
"""
st.session_state.generated_itinerary = full_paper
st.session_state.generation_complete = True
date_str = datetime.now().strftime("%Y-%m-%d")
st.session_state.filename = f"{user_input['paper_title'].replace(' ', '_')}_{date_str}_paper.txt"
if st.session_state.generation_complete:
st.markdown(f"""
<div class="modern-card">
<div style="text-align: center;">
<h2>{t("itinerary_ready")}</h2>
<p>{t("personalized_experience")}</p>
</div>
</div>
""", unsafe_allow_html=True)
# ํƒญ ์ƒ์„ฑ (์ „์ฒด ๋…ผ๋ฌธ, ์„ธ๋ถ€ ์ •๋ณด, ๋‹ค์šด๋กœ๋“œ/๊ณต์œ , ์‹œ๊ฐํ™”, ์ฑ—๋ด‡)
full_paper_tab, details_tab, download_tab, visualization_tab, chatbot_tab = st.tabs([
"๐Ÿ—’๏ธ " + t("full_itinerary"),
"๐Ÿ’ผ " + t("details"),
"๐Ÿ’พ " + t("download_share"),
"๐Ÿ“Š Visualization",
"๐Ÿค– ์ฑ—๋ด‡ ์ธํ„ฐํŽ˜์ด์Šค"
])
with full_paper_tab:
st.text_area("Your Research Paper", st.session_state.generated_itinerary, height=600)
with details_tab:
agent_tabs = st.tabs(["๐Ÿ“š Literature Review", "๐Ÿ“ Outline", "โœ๏ธ Draft", "๐Ÿ”— Citations", "๐Ÿ–‹๏ธ Edited Version"])
with agent_tabs[0]:
st.markdown("### Literature Review")
st.markdown(st.session_state.results.get("literature_review", ""))
with agent_tabs[1]:
st.markdown("### Outline")
st.markdown(st.session_state.results.get("outline", ""))
with agent_tabs[2]:
st.markdown("### Draft")
st.markdown(st.session_state.results.get("draft", ""))
with agent_tabs[3]:
st.markdown("### Citations")
st.markdown(st.session_state.results.get("citations", ""))
with agent_tabs[4]:
st.markdown("### Edited Version")
st.markdown(st.session_state.results.get("edited", ""))
with download_tab:
col1, col2 = st.columns([2, 1])
with col1:
st.markdown("### " + t("save_itinerary"))
st.markdown("Download your research paper to access it offline or share with your colleagues.")
st.markdown(f"""
<div style="background-color: #f8f9fa; padding: 15px; border-radius: 10px; margin-top: 20px;">
<h4>{t("your_itinerary_file")}</h4>
<p style="font-size: 0.9rem; color: #6c757d;">{t("text_format")}</p>
</div>
""", unsafe_allow_html=True)
st.markdown("<div>" + get_download_link(st.session_state.generated_itinerary, st.session_state.filename) + "</div>", unsafe_allow_html=True)
st.markdown("### " + t("share_itinerary"))
st.markdown("*Coming soon: Email your paper or share via social media.*")
with col2:
st.markdown("### " + t("save_for_mobile"))
st.markdown("*Coming soon: QR code for easy access on your phone*")
with visualization_tab:
st.markdown("### Visualization")
st.markdown("A conceptual diagram or visualization related to your research paper can be displayed here. (Feature under development)")
with chatbot_tab:
st.markdown("### AI ์ฑ—๋ด‡ ์ธํ„ฐํŽ˜์ด์Šค")
if "chat_history" not in st.session_state:
st.session_state.chat_history = []
user_message = st.text_input("๋ฉ”์‹œ์ง€๋ฅผ ์ž…๋ ฅํ•˜์„ธ์š”:", key="chat_input")
if st.button("์ „์†ก", key="send_button"):
if user_message:
response = run_task(chatbot_task, user_message)
st.session_state.chat_history.append({
"speaker": "์‚ฌ์šฉ์ž",
"message": user_message,
"time": datetime.now()
})
st.session_state.chat_history.append({
"speaker": "AI",
"message": response,
"time": datetime.now()
})
st.markdown("<div style='max-height:400px; overflow-y:auto; padding:10px; border:1px solid #eaeaea; border-radius:6px;'>", unsafe_allow_html=True)
for chat in st.session_state.chat_history:
time_str = chat["time"].strftime("%H:%M:%S")
st.markdown(f"**{chat['speaker']}** ({time_str}): {chat['message']}")
st.markdown("</div>", unsafe_allow_html=True)
st.markdown("""
<div style="text-align: center; padding: 20px; color: #6c757d; font-size: 0.8rem;">
<p>""" + t("built_with") + """</p>
</div>
""", unsafe_allow_html=True)