Spaces:
Sleeping
Sleeping
File size: 4,894 Bytes
400552c 190ecd4 400552c 78f9503 190ecd4 005a4c0 400552c 190ecd4 400552c 78f9503 190ecd4 78f9503 d2d3ca7 78f9503 190ecd4 78f9503 190ecd4 005a4c0 190ecd4 78f9503 005a4c0 78f9503 400552c 78f9503 e41e611 6ed422e 78f9503 9135c43 794fb98 6ed422e 78f9503 |
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 |
# Standard library imports
import os
import re
import sqlite3
from typing import List, Tuple, Optional
# Third-party imports
import gradio as gr
from dotenv import load_dotenv
# SmolaGents imports
from smolagents import CodeAgent, LiteLLMModel, tool
from smolagents.agent_types import AgentText
from smolagents.memory import ActionStep, TaskStep
# Local imports
from init_db import Project
# Load environment variables
load_dotenv()
@tool
def sql_engine(query: str) -> List[Tuple]:
"""
Allows you to perform SQL queries on the table in SQLite database. Returns list of results.
The table is named 'project'. Its description is as follows:
Columns:
- url: VARCHAR(255) - URL of the project
- title: VARCHAR(255) - Title of the project
- deposit: INTEGER - Percentage of payment before completion
- min_price: INTEGER - Lowest available apartment price in CZK with VAT
- status: VARCHAR(255) - Status of the project (preparation, selling, sold out)
- city: VARCHAR(255) - City of the project
- lat: FLOAT - GPS latitude coordinates
- lng: FLOAT - GPS longitude coordinates
- start_year: INTEGER - Year of construction start
- end_year: INTEGER - Year of construction end
- developer: VARCHAR(255) - Name of the construction company
- ignore: BOOLEAN - If True, the project does not have any apartments for sale
# Apartment prices by type (all INTEGER in CZK with VAT)
- price_1kk: Price of 1+kk apartment
- price_2kk: Price of 2+kk apartment
- price_3kk: Price of 3+kk apartment
- price_4kk: Price of 4+kk apartment
- price_5kk: Price of 5+kk apartment
- price_6kk: Price of 6+kk apartment
- price_7kk: Price of 7+kk apartment
- price_8kk: Price of 8+kk apartment
- price_9kk: Price of 9+kk apartment
- price_10kk: Price of 10+kk apartment
- price_1_1: Price of 1+1 apartment
- price_2_1: Price of 2+1 apartment
- price_3_1: Price of 3+1 apartment
- price_4_1: Price of 4+1 apartment
- price_5_1: Price of 5+1 apartment
- price_6_1: Price of 6+1 apartment
- price_7_1: Price of 7+1 apartment
- price_8_1: Price of 8+1 apartment
- price_9_1: Price of 9+1 apartment
- price_10_1: Price of 10+1 apartment
- content: TEXT - Raw content (ignore to save tokens)
- created_at: DATETIME - Date and time of creation
Args:
query: The query to perform. This should be correct SQL.
"""
con = sqlite3.connect("estate.db")
result = con.execute(query)
all_rows = result.fetchall()
con.close()
return all_rows
sql_distance_query = """
SELECT
id,
url,
structure->>'$.lat' AS lat,
structure->>'$.lng' AS lng,
(6371000 * acos(
cos(radians(50.08804)) * cos(radians(CAST(structure->>'$.lat' AS FLOAT))) *
cos(radians(CAST(structure->>'$.lng' AS FLOAT)) - radians(14.42076)) +
sin(radians(50.08804)) * sin(radians(CAST(structure->>'$.lat' AS FLOAT)))
)) AS distance
FROM project
WHERE structure->>'$.lat' IS NOT NULL
AND structure->>'$.lng' IS NOT NULL
ORDER BY distance ASC
LIMIT 5;
"""
litemodel = LiteLLMModel("o3-mini-2025-01-31", api_base="https://api.openai.com/v1", api_key=os.getenv("OPENAI_API_KEY"))
codeagent = CodeAgent(tools=[sql_engine], model=litemodel)
def chat_with_sql(message, history):
try:
input_message = "You are a helpful assistant that presents information about real estate projects in database for investors. Use 'IS NOT NULL' when necessary. Translate any city names from input into Czech language. Present useful fields such as url. Do not make up information or hallucinate. If final result rows are empty respond that you could not find records matching criteria. Use only Czech city names. Note that there is limit 10 records for database result. Reformat final answer in markdown. User Input:\n\n {message}"
prompt = input_message.format(message=message)
result = codeagent.run(prompt, reset=False)
yield gr.ChatMessage(role="assistant", content=result)
except Exception as e:
yield gr.ChatMessage(role="assistant", content=f"Error: {str(e)}")
# Create the Gradio interface
count_projects = Project.select().where(Project.structure.is_null(False)).count()
demo = gr.ChatInterface(
fn=chat_with_sql,
title="Estate Chat",
description=f"Planning to buy a new apartment? Ask me about {count_projects} new constructions (novostavby) in Prague 🇨🇿",
examples=[
"Cheapest apartment in Prague",
"Project with the lowest deposit before completion",
"All 2+kk below 8000000 CZK",
],
type="messages"
)
if __name__ == "__main__":
demo.launch(share=False, debug=True) |