Spaces:
Sleeping
Sleeping
File size: 5,889 Bytes
6a0ec6a 207d5a0 6a0ec6a 1767e22 5a55ea7 6349cf9 5a55ea7 28200f6 5a55ea7 1767e22 5a55ea7 28200f6 5a55ea7 20e319d 5a55ea7 28200f6 5a55ea7 28200f6 20e319d 5a55ea7 20e319d 5a55ea7 20e319d 5a55ea7 20e319d 5a55ea7 20e319d 207d5a0 042246b 28200f6 207d5a0 28200f6 207d5a0 28200f6 042246b 5a55ea7 7306c07 28200f6 5a55ea7 6349cf9 5a55ea7 61d9b40 5a55ea7 2443195 5a55ea7 61d9b40 5a55ea7 f8c651a 61d9b40 5a55ea7 1f7ee11 5a55ea7 215368b 28200f6 5a55ea7 edb7e14 5a55ea7 28200f6 5a55ea7 edb7e14 5a55ea7 6349cf9 1f7ee11 5a55ea7 1f7ee11 1df3c5d 1f7ee11 5a55ea7 1767e22 5a55ea7 1767e22 5a55ea7 1767e22 5a55ea7 6a0ec6a 0380e03 6349cf9 |
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 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 |
import os
import gradio as gr
from sqlalchemy import text, inspect
from smolagents import tool, CodeAgent, HfApiModel
import pandas as pd
import tempfile
from database import engine, initialize_database
# Ensure the database initializes
initialize_database()
# Function to execute SQL script from uploaded file
def execute_sql_script(file_path):
"""
Executes an uploaded SQL file to initialize the database.
Args:
file_path (str): Path to the SQL file.
Returns:
str: Success message or error description.
"""
try:
with engine.connect() as con:
with open(file_path, "r") as f:
sql_script = f.read()
con.execute(text(sql_script))
return "SQL file executed successfully."
except Exception as e:
return f"Error: {str(e)}"
# Function to fetch table names dynamically
def get_table_names():
"""
Returns a list of all tables in the database.
Returns:
list: List of table names.
"""
inspector = inspect(engine)
return inspector.get_table_names()
# Function to fetch table schema dynamically
def get_table_schema(table_name):
"""
Returns a list of column names for a given table.
Args:
table_name (str): Name of the table.
Returns:
list: List of column names.
"""
inspector = inspect(engine)
columns = inspector.get_columns(table_name)
return [col["name"] for col in columns]
# Function to fetch table data dynamically
def get_table_data(table_name):
"""
Retrieves all rows from the specified table as a Pandas DataFrame.
Args:
table_name (str): Name of the table.
Returns:
pd.DataFrame: Table data or an error message.
"""
try:
with engine.connect() as con:
result = con.execute(text(f"SELECT * FROM {table_name}"))
rows = result.fetchall()
columns = get_table_schema(table_name)
if not rows:
return pd.DataFrame(columns=columns)
return pd.DataFrame(rows, columns=columns)
except Exception as e:
return pd.DataFrame({"Error": [str(e)]})
# SQL Execution Tool (FIXED Docstring for `smolagents`)
@tool
def sql_engine(query: str) -> str:
"""
Executes an SQL SELECT query and returns the results.
Args:
query (str): The SQL query string to execute.
Returns:
str: The formatted query results, or an error message if the query fails.
"""
try:
with engine.connect() as con:
rows = con.execute(text(query)).fetchall()
if not rows:
return "No results found."
return "\n".join([", ".join(map(str, row)) for row in rows])
except Exception as e:
return f"Error: {str(e)}"
# Function to generate and execute SQL queries dynamically
def query_sql(user_query: str) -> str:
"""
Processes a user’s natural language query and generates an SQL query dynamically.
Args:
user_query (str): The question asked by the user.
Returns:
str: SQL query results or an error message.
"""
tables = get_table_names()
if not tables:
return "Error: No tables found. Please upload an SQL file first."
schema_info = "Available tables and columns:\n"
for table in tables:
columns = get_table_schema(table)
schema_info += f"Table '{table}' has columns: {', '.join(columns)}.\n"
schema_info += "Generate a valid SQL SELECT query using ONLY these column names. DO NOT return anything other than the SQL query itself."
generated_sql = agent.run(f"{schema_info} Convert this request into SQL: {user_query}")
if not isinstance(generated_sql, str) or not generated_sql.strip().lower().startswith(("select", "show", "pragma")):
return "Error: Only SELECT queries are allowed."
return sql_engine(generated_sql)
# Function to handle query input
def handle_query(user_input: str) -> str:
"""
Handles user input and returns the SQL query result.
Args:
user_input (str): User's natural language query.
Returns:
str: The query result or error message.
"""
return query_sql(user_input)
# Function to handle SQL file uploads
def handle_file_upload(file):
"""
Handles file upload, executes SQL, and updates database schema dynamically.
Args:
file (File): Uploaded SQL file.
Returns:
tuple: Execution result message and updated table data.
"""
temp_file_path = tempfile.mkstemp(suffix=".sql")[1]
with open(temp_file_path, "wb") as temp_file:
temp_file.write(file.read())
result = execute_sql_script(temp_file_path)
tables = get_table_names()
if tables:
table_data = {table: get_table_data(table) for table in tables}
else:
table_data = {"Error": ["No tables found after execution. Ensure your SQL file creates tables."]}
return result, table_data
# Initialize CodeAgent for SQL query generation
agent = CodeAgent(
tools=[sql_engine],
model=HfApiModel(model_id="Qwen/Qwen2.5-Coder-32B-Instruct"),
)
# Gradio UI
with gr.Blocks() as demo:
gr.Markdown("## SQL Query Interface")
with gr.Row():
user_input = gr.Textbox(label="Ask a question about the data")
query_output = gr.Textbox(label="Result")
user_input.change(fn=handle_query, inputs=user_input, outputs=query_output)
gr.Markdown("## Upload SQL File to Execute")
file_upload = gr.File(label="Upload SQL File")
upload_output = gr.Textbox(label="Execution Result")
# Dynamic table display
table_output = gr.Dataframe(label="Database Tables (Dynamic)")
file_upload.change(fn=handle_file_upload, inputs=file_upload, outputs=[upload_output, table_output])
if __name__ == "__main__":
demo.launch(server_name="0.0.0.0", server_port=7860, share=True)
|