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)