Spaces:
Sleeping
Sleeping
Upload 15 files
Browse files- .gitattributes +2 -0
- .gitignore +6 -0
- Gemini_agents.ipynb +1232 -0
- README.md +91 -12
- agents.py +253 -0
- app.py +44 -0
- calendar_tools.py +236 -0
- chroma/caico/chroma.sqlite3 +3 -0
- chroma/natal/chroma.sqlite3 +3 -0
- chroma/pipa/chroma.sqlite3 +0 -0
- get_embedding_function.py +6 -0
- google_apis.py +43 -0
- planing_tools.py +89 -0
- populate_database.py +134 -0
- requirements.txt +18 -0
- video_transcriptor.py +95 -0
.gitattributes
CHANGED
@@ -33,3 +33,5 @@ saved_model/**/* filter=lfs diff=lfs merge=lfs -text
|
|
33 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
34 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
35 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
|
|
|
|
|
33 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
34 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
35 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
36 |
+
chroma/caico/chroma.sqlite3 filter=lfs diff=lfs merge=lfs -text
|
37 |
+
chroma/natal/chroma.sqlite3 filter=lfs diff=lfs merge=lfs -text
|
.gitignore
ADDED
@@ -0,0 +1,6 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
.env
|
2 |
+
.venv
|
3 |
+
__pycache__
|
4 |
+
token files
|
5 |
+
client_secret.json
|
6 |
+
pasta_com_videos
|
Gemini_agents.ipynb
ADDED
@@ -0,0 +1,1232 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{
|
2 |
+
"cells": [
|
3 |
+
{
|
4 |
+
"cell_type": "code",
|
5 |
+
"execution_count": null,
|
6 |
+
"metadata": {
|
7 |
+
"colab": {
|
8 |
+
"base_uri": "https://localhost:8080/"
|
9 |
+
},
|
10 |
+
"id": "4gIXtYDb4o3o",
|
11 |
+
"outputId": "b74b0d76-37d0-427c-c8fd-fb9a9337d061"
|
12 |
+
},
|
13 |
+
"outputs": [
|
14 |
+
{
|
15 |
+
"name": "stderr",
|
16 |
+
"output_type": "stream",
|
17 |
+
"text": [
|
18 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\tqdm\\auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
|
19 |
+
" from .autonotebook import tqdm as notebook_tqdm\n"
|
20 |
+
]
|
21 |
+
}
|
22 |
+
],
|
23 |
+
"source": [
|
24 |
+
"import os\n",
|
25 |
+
"\n",
|
26 |
+
"from langchain.agents import Tool\n",
|
27 |
+
"from langchain.memory import ChatMessageHistory, ConversationBufferWindowMemory\n",
|
28 |
+
"from langchain_google_genai import (\n",
|
29 |
+
" ChatGoogleGenerativeAI,\n",
|
30 |
+
" HarmBlockThreshold,\n",
|
31 |
+
" HarmCategory,\n",
|
32 |
+
")\n",
|
33 |
+
"from langchain.utilities import DuckDuckGoSearchAPIWrapper\n",
|
34 |
+
"from langchain.agents import AgentExecutor\n",
|
35 |
+
"from langchain import hub\n",
|
36 |
+
"from langchain.agents.format_scratchpad import format_log_to_str\n",
|
37 |
+
"from langchain.agents.output_parsers import ReActSingleInputOutputParser\n",
|
38 |
+
"from langchain.tools.render import render_text_description\n",
|
39 |
+
"from datetime import datetime\n",
|
40 |
+
"from dotenv import load_dotenv"
|
41 |
+
]
|
42 |
+
},
|
43 |
+
{
|
44 |
+
"cell_type": "code",
|
45 |
+
"execution_count": null,
|
46 |
+
"metadata": {},
|
47 |
+
"outputs": [],
|
48 |
+
"source": [
|
49 |
+
"hoje = datetime.today()\n",
|
50 |
+
"\n",
|
51 |
+
"data_formatada = hoje.strftime(\"%Y/%m/%d\")\n",
|
52 |
+
"# Pega o dia da semana em inglês\n",
|
53 |
+
"dia_da_semana_ingles = hoje.strftime(\"%A\")\n",
|
54 |
+
"\n",
|
55 |
+
"# Dicionário para mapear os dias da semana de inglês para português\n",
|
56 |
+
"dias_da_semana_pt = {\n",
|
57 |
+
" \"Monday\": \"segunda-feira\",\n",
|
58 |
+
" \"Tuesday\": \"terça-feira\",\n",
|
59 |
+
" \"Wednesday\": \"quarta-feira\",\n",
|
60 |
+
" \"Thursday\": \"quinta-feira\",\n",
|
61 |
+
" \"Friday\": \"sexta-feira\",\n",
|
62 |
+
" \"Saturday\": \"sábado\",\n",
|
63 |
+
" \"Sunday\": \"domingo\"\n",
|
64 |
+
"}\n",
|
65 |
+
"\n",
|
66 |
+
"# Traduz o dia da semana para português\n",
|
67 |
+
"dia_da_semana_pt = dias_da_semana_pt.get(dia_da_semana_ingles, dia_da_semana_ingles)\n",
|
68 |
+
"\n",
|
69 |
+
"# Formata a data para utilizar no prompt\n",
|
70 |
+
"data_atual = f\"{dia_da_semana_pt}, {data_formatada}\""
|
71 |
+
]
|
72 |
+
},
|
73 |
+
{
|
74 |
+
"cell_type": "code",
|
75 |
+
"execution_count": null,
|
76 |
+
"metadata": {
|
77 |
+
"id": "oRYlHV3fy_Iq"
|
78 |
+
},
|
79 |
+
"outputs": [],
|
80 |
+
"source": [
|
81 |
+
"#(https://makersuite.google.com/)\n",
|
82 |
+
"\n",
|
83 |
+
"load_dotenv()\n",
|
84 |
+
"os.environ[\"GOOGLE_API_KEY\"] = os.getenv('GOOGLE_API_KEY')\n",
|
85 |
+
"os.environ[\"LANGCHAIN_API_KEY\"] = os.getenv('LANGCHAIN_API_KEY')"
|
86 |
+
]
|
87 |
+
},
|
88 |
+
{
|
89 |
+
"cell_type": "code",
|
90 |
+
"execution_count": 4,
|
91 |
+
"metadata": {
|
92 |
+
"id": "FEDgMFo84iGY"
|
93 |
+
},
|
94 |
+
"outputs": [],
|
95 |
+
"source": [
|
96 |
+
"llm = ChatGoogleGenerativeAI(\n",
|
97 |
+
" model=\"gemini-1.5-flash\",\n",
|
98 |
+
" convert_system_message_to_human=True,\n",
|
99 |
+
" handle_parsing_errors=True,\n",
|
100 |
+
" temperature=0.6,\n",
|
101 |
+
" max_tokens= 1000,\n",
|
102 |
+
" safety_settings = {\n",
|
103 |
+
" HarmCategory.HARM_CATEGORY_HARASSMENT: HarmBlockThreshold.BLOCK_ONLY_HIGH,\n",
|
104 |
+
" HarmCategory.HARM_CATEGORY_HATE_SPEECH: HarmBlockThreshold.BLOCK_ONLY_HIGH,\n",
|
105 |
+
" HarmCategory.HARM_CATEGORY_SEXUALLY_EXPLICIT: HarmBlockThreshold.BLOCK_ONLY_HIGH,\n",
|
106 |
+
" HarmCategory.HARM_CATEGORY_DANGEROUS_CONTENT: HarmBlockThreshold.BLOCK_ONLY_HIGH,\n",
|
107 |
+
" },\n",
|
108 |
+
")"
|
109 |
+
]
|
110 |
+
},
|
111 |
+
{
|
112 |
+
"cell_type": "code",
|
113 |
+
"execution_count": 5,
|
114 |
+
"metadata": {},
|
115 |
+
"outputs": [],
|
116 |
+
"source": [
|
117 |
+
"def transfer_to_calendar_agent(input_str):\n",
|
118 |
+
" return calendar_agent_executor.invoke({\"input\": input_str})\n",
|
119 |
+
"def transfer_to_travel_agent(input_str):\n",
|
120 |
+
" return travel_agent_executor.invoke({\"input\": input_str})"
|
121 |
+
]
|
122 |
+
},
|
123 |
+
{
|
124 |
+
"cell_type": "code",
|
125 |
+
"execution_count": 7,
|
126 |
+
"metadata": {
|
127 |
+
"id": "EY14yxCD4OzY"
|
128 |
+
},
|
129 |
+
"outputs": [],
|
130 |
+
"source": [
|
131 |
+
"from planing_tools import weatherapi_forecast_periods, query_rag\n",
|
132 |
+
"ddg_search = DuckDuckGoSearchAPIWrapper()\n",
|
133 |
+
"\n",
|
134 |
+
"travel_planing_tools = [\n",
|
135 |
+
" Tool(\n",
|
136 |
+
" name=\"DuckDuckGo Search\",\n",
|
137 |
+
" func=ddg_search.run,\n",
|
138 |
+
" description=\"\"\"Essa ferramenta DEVE ser utilizada para buscar eventos relevantes no período fornecido pelo usuário. \n",
|
139 |
+
" Ela é útil para obter informações sobre eventos ou atividades especiais que estão acontecendo na cidade de destino nas datas que o usuário informou. \n",
|
140 |
+
" O modelo deve usá-la para complementar as sugestões de atividades.\"\"\"\n",
|
141 |
+
" ),\n",
|
142 |
+
" Tool(\n",
|
143 |
+
" name=\"Weather Forecast\",\n",
|
144 |
+
" func=weatherapi_forecast_periods,\n",
|
145 |
+
" description=\"\"\"Essa ferramenta DEVE ser usada *antes* de gerar o roteiro turístico, após o modelo coletar todas as informações necessárias do usuário, incluindo as datas exatas. \n",
|
146 |
+
" O modelo deve consultar o clima separadamente para cada dia do período informado, garantindo que as atividades do roteiro sejam planejadas de acordo com as condições climáticas. \n",
|
147 |
+
" A previsão do tempo é um passo obrigatório quando é um fator relevante para o planejamento.\"\"\"\n",
|
148 |
+
" ),\n",
|
149 |
+
" Tool(\n",
|
150 |
+
" name=\"Query RAG\",\n",
|
151 |
+
" func=query_rag,\n",
|
152 |
+
" description=\"\"\"Esta ferramenta deve ser usada quando o modelo souber a cidade de destino e os interesses do usuário, com o objetivo de fornecer informações sobre pontos turísticos e atrações que se alinham com esses interesses. \n",
|
153 |
+
" O modelo deve utilizar essa ferramenta para sugerir atividades e lugares específicos a visitar, baseados na cidade e nos interesses fornecidos.\"\"\"\n",
|
154 |
+
" ),\n",
|
155 |
+
" Tool(\n",
|
156 |
+
" name=\"Calendar Agent\",\n",
|
157 |
+
" func=transfer_to_calendar_agent,\n",
|
158 |
+
" description=\"Esse agente lida com tudo relacionado ao calendário. As mensagens enviadas a ele devem estar em Português.\"\n",
|
159 |
+
" ),\n",
|
160 |
+
"]"
|
161 |
+
]
|
162 |
+
},
|
163 |
+
{
|
164 |
+
"cell_type": "code",
|
165 |
+
"execution_count": 8,
|
166 |
+
"metadata": {},
|
167 |
+
"outputs": [],
|
168 |
+
"source": [
|
169 |
+
"from calendar_tools import list_calendar_list, list_calendar_events, insert_calendar_event, create_calendar\n",
|
170 |
+
"\n",
|
171 |
+
"google_calendar_tools = [\n",
|
172 |
+
" Tool(\n",
|
173 |
+
" name=\"Create Calendar List\",\n",
|
174 |
+
" func=create_calendar,\n",
|
175 |
+
" description=\"\"\"\n",
|
176 |
+
" Use a função create_calendar_list para criar um novo calendário.\n",
|
177 |
+
" Forneça o resumo (título) do calendário que você deseja criar.\n",
|
178 |
+
" - Exemplo de uso: 'Meu Calendário de Viagens'\n",
|
179 |
+
" - Essa função vai criar um novo calendário com o resumo fornecido.\n",
|
180 |
+
" \"\"\"\n",
|
181 |
+
" ),\n",
|
182 |
+
" Tool(\n",
|
183 |
+
" name=\"List Calendar List\",\n",
|
184 |
+
" func=list_calendar_list,\n",
|
185 |
+
" description=\"\"\"\n",
|
186 |
+
" Use a função list_calendar_list para obter a lista dos seus calendários disponíveis no Google Calendar.\n",
|
187 |
+
" Se você quiser limitar o número de calendários retornados, forneça o número máximo desejado.\n",
|
188 |
+
" - Exemplo de uso: Forneça o número máximo de calendários a serem listados, como 50. Se não quiser um limite, apenas diga para listar 200 o máximo por padrão.\n",
|
189 |
+
" \"\"\"\n",
|
190 |
+
" ),\n",
|
191 |
+
" Tool(\n",
|
192 |
+
" name=\"List Calendar Events\",\n",
|
193 |
+
" func=list_calendar_events,\n",
|
194 |
+
" description=\"\"\"\n",
|
195 |
+
" Use a função list_calendar_events para obter a lista de eventos de um calendário específico.\n",
|
196 |
+
" Você precisa fornecer o ID do calendário e, opcionalmente, o número máximo de eventos a serem listados.\n",
|
197 |
+
" - Exemplo de uso para o calendário principal: forneça 'primary' como o ID do calendário.\n",
|
198 |
+
" - Exemplo de uso com limite: forneça o ID do calendário e o número máximo de eventos, como 'primary', 20.\n",
|
199 |
+
" - Para listar eventos de um calendário específico, primeiro liste seus calendários para encontrar o ID correto.\n",
|
200 |
+
" \"\"\"\n",
|
201 |
+
" ),\n",
|
202 |
+
" Tool(\n",
|
203 |
+
" name=\"Insert Calendar Event\",\n",
|
204 |
+
" func=insert_calendar_event,\n",
|
205 |
+
" description=\"\"\"\n",
|
206 |
+
" Use a função Insert Calendar Event para adicionar **um único evento** a um calendário Google.\n",
|
207 |
+
" Esta função deve ser chamada **uma vez para cada evento** que precisa ser agendado.\n",
|
208 |
+
"\n",
|
209 |
+
" A entrada para esta função deve ser uma descrição clara e completa de **um único evento**\n",
|
210 |
+
" em linguagem natural, contendo todas as informações necessárias para agendá-lo individualmente.\n",
|
211 |
+
"\n",
|
212 |
+
" Se o usuário pediu para criar um novo calendário, você DEVE utilizar o calendar_id desse novo calendário para a inserção dos eventos. \n",
|
213 |
+
" \n",
|
214 |
+
" **Formato de entrada esperado:**\n",
|
215 |
+
"\n",
|
216 |
+
" **Descrição em Linguagem Natural (obrigatório):**\n",
|
217 |
+
" Forneça uma descrição textual detalhada para **um único evento**. Inclua o calendar_id (se fornecido pelo usuário), o que será feito,\n",
|
218 |
+
" quando (data e hora específicas), onde (local exato) e quaisquer detalhes adicionais relevantes\n",
|
219 |
+
" para esse evento em particular.\n",
|
220 |
+
"\n",
|
221 |
+
" **Exemplos de descrições de eventos individuais:**\n",
|
222 |
+
" - \"Agendar almoço com a Maria na sexta-feira ao meio-dia no restaurante X.\"\n",
|
223 |
+
" - \"Criar um compromisso para pagar as contas no dia 10 de janeiro às 9h da manhã.\"\n",
|
224 |
+
" - \"Bloquear minha agenda para foco no projeto Y na próxima segunda-feira, das 14h às 17h, na minha mesa.\"\n",
|
225 |
+
"\n",
|
226 |
+
" **Como usar esta função corretamente:**\n",
|
227 |
+
"\n",
|
228 |
+
" 1. **Identifique cada evento individualmente:** Se a intenção é agendar múltiplos eventos,\n",
|
229 |
+
" trate cada um como uma chamada separada para esta função.\n",
|
230 |
+
" 2. **Obtenha o ID do calendário (se necessário):** Se o usuário mencionar um nome de\n",
|
231 |
+
" calendário específico, use a função 'list_calendar_list' **primeiro** para obter\n",
|
232 |
+
" o 'calendar_id' correspondente.\n",
|
233 |
+
" 3. **Chame esta função com a descrição de um único evento:** Forneça a descrição completa\n",
|
234 |
+
" em linguagem natural para o evento específico que você deseja agendar.\n",
|
235 |
+
"\n",
|
236 |
+
" **Informações essenciais para cada descrição de evento:**\n",
|
237 |
+
" - **O que:** Uma descrição concisa do evento (ex: \"Reunião de equipe\", \"Consulta médica\").\n",
|
238 |
+
" - **Quando:** A data e hora de início e fim **precisas** do evento.\n",
|
239 |
+
" - **Onde:** O local **específico** do evento.\n",
|
240 |
+
" - **Fuso Horário (opcional):** Se aplicável, mencione o fuso horário se for diferente\n",
|
241 |
+
" de \"America/Fortaleza\".\n",
|
242 |
+
"\n",
|
243 |
+
" **Exemplo de uso**\n",
|
244 |
+
"\n",
|
245 |
+
" *Usuário: Agendar visita ao Aquário Natal sábado das 9h às 12h.*\n",
|
246 |
+
" *Agente:* (Chama 'Insert Calendar Event' com a Action Input: \"Agendar visita ao Aquário Natal sábado das 9h às 12h.\")\n",
|
247 |
+
"\n",
|
248 |
+
" **Observação IMPORTANTE:** Esta função **NÃO** deve ser usada para criar múltiplos eventos\n",
|
249 |
+
" com uma única chamada. Cada chamada deve corresponder a **um único evento**.\n",
|
250 |
+
"\n",
|
251 |
+
" Ao final da execução desta função, você DEVE fornecer ao usuário a confirmação do agendamento\n",
|
252 |
+
" desse evento específico, incluindo o título, data, hora e link do evento criado (se disponível).\n",
|
253 |
+
" \"\"\"\n",
|
254 |
+
" )\n",
|
255 |
+
"]"
|
256 |
+
]
|
257 |
+
},
|
258 |
+
{
|
259 |
+
"cell_type": "code",
|
260 |
+
"execution_count": null,
|
261 |
+
"metadata": {
|
262 |
+
"colab": {
|
263 |
+
"base_uri": "https://localhost:8080/",
|
264 |
+
"height": 209
|
265 |
+
},
|
266 |
+
"id": "FP7Q3SBS5qb3",
|
267 |
+
"outputId": "005c96da-55f7-401c-b00a-4a053dcebaf8"
|
268 |
+
},
|
269 |
+
"outputs": [],
|
270 |
+
"source": [
|
271 |
+
"travel_planing_prompt = hub.pull(\"tales/agente_turismo\")\n",
|
272 |
+
"google_calendar_prompt = hub.pull(\"tales/agente_calendario\")"
|
273 |
+
]
|
274 |
+
},
|
275 |
+
{
|
276 |
+
"cell_type": "code",
|
277 |
+
"execution_count": null,
|
278 |
+
"metadata": {
|
279 |
+
"id": "MuPTv06R8EOH"
|
280 |
+
},
|
281 |
+
"outputs": [],
|
282 |
+
"source": [
|
283 |
+
"llm_with_stop = llm.bind(stop=[\"\\nObservation\"])\n",
|
284 |
+
"planing_prompt = travel_planing_prompt.partial(\n",
|
285 |
+
" tools=render_text_description(travel_planing_tools),\n",
|
286 |
+
" tool_names=\", \".join([t.name for t in travel_planing_tools]),\n",
|
287 |
+
")"
|
288 |
+
]
|
289 |
+
},
|
290 |
+
{
|
291 |
+
"cell_type": "code",
|
292 |
+
"execution_count": 11,
|
293 |
+
"metadata": {},
|
294 |
+
"outputs": [],
|
295 |
+
"source": [
|
296 |
+
"calendar_prompt = google_calendar_prompt.partial(\n",
|
297 |
+
" tools=render_text_description(google_calendar_tools),\n",
|
298 |
+
" tool_names=\", \".join([t.name for t in google_calendar_tools]),\n",
|
299 |
+
")"
|
300 |
+
]
|
301 |
+
},
|
302 |
+
{
|
303 |
+
"cell_type": "code",
|
304 |
+
"execution_count": 12,
|
305 |
+
"metadata": {
|
306 |
+
"colab": {
|
307 |
+
"base_uri": "https://localhost:8080/"
|
308 |
+
},
|
309 |
+
"id": "LHGz2W5g4cSx",
|
310 |
+
"outputId": "fe5fbb96-6f6d-48ce-ba79-63d1b9a732d1"
|
311 |
+
},
|
312 |
+
"outputs": [
|
313 |
+
{
|
314 |
+
"name": "stderr",
|
315 |
+
"output_type": "stream",
|
316 |
+
"text": [
|
317 |
+
"C:\\Users\\Tales\\AppData\\Local\\Temp\\ipykernel_5320\\275711510.py:2: LangChainDeprecationWarning: Please see the migration guide at: https://python.langchain.com/docs/versions/migrating_memory/\n",
|
318 |
+
" memory = ConversationBufferWindowMemory(k=20, chat_memory=history, memory_key=\"chat_history\")\n"
|
319 |
+
]
|
320 |
+
}
|
321 |
+
],
|
322 |
+
"source": [
|
323 |
+
"history = ChatMessageHistory()\n",
|
324 |
+
"memory = ConversationBufferWindowMemory(k=20, chat_memory=history, memory_key=\"chat_history\")"
|
325 |
+
]
|
326 |
+
},
|
327 |
+
{
|
328 |
+
"cell_type": "code",
|
329 |
+
"execution_count": 13,
|
330 |
+
"metadata": {
|
331 |
+
"id": "zuQgO-EM9mdE"
|
332 |
+
},
|
333 |
+
"outputs": [],
|
334 |
+
"source": [
|
335 |
+
"travel_planing_agent = (\n",
|
336 |
+
" {\n",
|
337 |
+
" \"input\": lambda x: x[\"input\"],\n",
|
338 |
+
" \"agent_scratchpad\": lambda x: format_log_to_str(x[\"intermediate_steps\"]),\n",
|
339 |
+
" \"chat_history\": lambda x: x[\"chat_history\"],\n",
|
340 |
+
" \"data_atual\": lambda x: data_atual,\n",
|
341 |
+
" }\n",
|
342 |
+
" | planing_prompt\n",
|
343 |
+
" | llm_with_stop\n",
|
344 |
+
" | ReActSingleInputOutputParser()\n",
|
345 |
+
")\n",
|
346 |
+
"\n",
|
347 |
+
"travel_agent_executor = AgentExecutor(agent=travel_planing_agent, tools=travel_planing_tools, verbose=True, memory=memory)"
|
348 |
+
]
|
349 |
+
},
|
350 |
+
{
|
351 |
+
"cell_type": "code",
|
352 |
+
"execution_count": 14,
|
353 |
+
"metadata": {},
|
354 |
+
"outputs": [],
|
355 |
+
"source": [
|
356 |
+
"google_calendar_agent = (\n",
|
357 |
+
" {\n",
|
358 |
+
" \"input\": lambda x: x[\"input\"],\n",
|
359 |
+
" \"agent_scratchpad\": lambda x: format_log_to_str(x[\"intermediate_steps\"]),\n",
|
360 |
+
" \"chat_history\": lambda x: x[\"chat_history\"],\n",
|
361 |
+
" \"data_atual\": lambda x: data_atual,\n",
|
362 |
+
" }\n",
|
363 |
+
" | calendar_prompt\n",
|
364 |
+
" | llm_with_stop\n",
|
365 |
+
" | ReActSingleInputOutputParser()\n",
|
366 |
+
")\n",
|
367 |
+
"\n",
|
368 |
+
"calendar_agent_executor = AgentExecutor(agent=google_calendar_agent, tools=google_calendar_tools, verbose=True, memory=memory)"
|
369 |
+
]
|
370 |
+
},
|
371 |
+
{
|
372 |
+
"cell_type": "code",
|
373 |
+
"execution_count": 15,
|
374 |
+
"metadata": {},
|
375 |
+
"outputs": [
|
376 |
+
{
|
377 |
+
"name": "stdout",
|
378 |
+
"output_type": "stream",
|
379 |
+
"text": [
|
380 |
+
"\n",
|
381 |
+
"\n",
|
382 |
+
"\u001b[1m> Entering new AgentExecutor chain...\u001b[0m\n"
|
383 |
+
]
|
384 |
+
},
|
385 |
+
{
|
386 |
+
"name": "stderr",
|
387 |
+
"output_type": "stream",
|
388 |
+
"text": [
|
389 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
390 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
391 |
+
]
|
392 |
+
},
|
393 |
+
{
|
394 |
+
"name": "stdout",
|
395 |
+
"output_type": "stream",
|
396 |
+
"text": [
|
397 |
+
"\u001b[32;1m\u001b[1;3mThought: Do I need to use a tool? No\n",
|
398 |
+
"Final Answer: Boa noite! Para começarmos a planejar sua viagem, preciso de algumas informações. Para onde você está planejando viajar?\n",
|
399 |
+
"\u001b[0m\n",
|
400 |
+
"\n",
|
401 |
+
"\u001b[1m> Finished chain.\u001b[0m\n",
|
402 |
+
"Boa noite! Para começarmos a planejar sua viagem, preciso de algumas informações. Para onde você está planejando viajar?\n"
|
403 |
+
]
|
404 |
+
}
|
405 |
+
],
|
406 |
+
"source": [
|
407 |
+
"response = travel_agent_executor.invoke({\"input\": 'Boa noite'})\n",
|
408 |
+
"\n",
|
409 |
+
"print(response[\"output\"])"
|
410 |
+
]
|
411 |
+
},
|
412 |
+
{
|
413 |
+
"cell_type": "code",
|
414 |
+
"execution_count": 16,
|
415 |
+
"metadata": {},
|
416 |
+
"outputs": [
|
417 |
+
{
|
418 |
+
"name": "stdout",
|
419 |
+
"output_type": "stream",
|
420 |
+
"text": [
|
421 |
+
"\n",
|
422 |
+
"\n",
|
423 |
+
"\u001b[1m> Entering new AgentExecutor chain...\u001b[0m\n"
|
424 |
+
]
|
425 |
+
},
|
426 |
+
{
|
427 |
+
"name": "stderr",
|
428 |
+
"output_type": "stream",
|
429 |
+
"text": [
|
430 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
431 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
432 |
+
]
|
433 |
+
},
|
434 |
+
{
|
435 |
+
"name": "stdout",
|
436 |
+
"output_type": "stream",
|
437 |
+
"text": [
|
438 |
+
"\u001b[32;1m\u001b[1;3mThought: Do I need to use a tool? No\n",
|
439 |
+
"Final Answer: Ótimo! Natal é uma excelente escolha. Quais atividades ou experiências você está buscando nessa viagem? Por exemplo, prefere explorar praias, restaurantes, eventos, museus, história, cultura, vida noturna, natureza, ou algo mais?\n",
|
440 |
+
"\u001b[0m\n",
|
441 |
+
"\n",
|
442 |
+
"\u001b[1m> Finished chain.\u001b[0m\n",
|
443 |
+
"Ótimo! Natal é uma excelente escolha. Quais atividades ou experiências você está buscando nessa viagem? Por exemplo, prefere explorar praias, restaurantes, eventos, museus, história, cultura, vida noturna, natureza, ou algo mais?\n"
|
444 |
+
]
|
445 |
+
}
|
446 |
+
],
|
447 |
+
"source": [
|
448 |
+
"response = travel_agent_executor.invoke({\"input\": 'Eu estou planejando visitar Natal no Rio Grande do Norte'})\n",
|
449 |
+
"\n",
|
450 |
+
"print(response[\"output\"])"
|
451 |
+
]
|
452 |
+
},
|
453 |
+
{
|
454 |
+
"cell_type": "code",
|
455 |
+
"execution_count": 17,
|
456 |
+
"metadata": {},
|
457 |
+
"outputs": [
|
458 |
+
{
|
459 |
+
"name": "stdout",
|
460 |
+
"output_type": "stream",
|
461 |
+
"text": [
|
462 |
+
"\n",
|
463 |
+
"\n",
|
464 |
+
"\u001b[1m> Entering new AgentExecutor chain...\u001b[0m\n"
|
465 |
+
]
|
466 |
+
},
|
467 |
+
{
|
468 |
+
"name": "stderr",
|
469 |
+
"output_type": "stream",
|
470 |
+
"text": [
|
471 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
472 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
473 |
+
]
|
474 |
+
},
|
475 |
+
{
|
476 |
+
"name": "stdout",
|
477 |
+
"output_type": "stream",
|
478 |
+
"text": [
|
479 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
480 |
+
"Thought: Do I need to use a tool? No\n",
|
481 |
+
"Final Answer: Perfeito! Praias e restaurantes regionais em Natal. Tem um orçamento em mente ou um limite de gastos que gostaria de seguir durante a viagem?\n",
|
482 |
+
"```\n",
|
483 |
+
"\u001b[0m\n",
|
484 |
+
"\n",
|
485 |
+
"\u001b[1m> Finished chain.\u001b[0m\n",
|
486 |
+
"Perfeito! Praias e restaurantes regionais em Natal. Tem um orçamento em mente ou um limite de gastos que gostaria de seguir durante a viagem?\n",
|
487 |
+
"```\n"
|
488 |
+
]
|
489 |
+
}
|
490 |
+
],
|
491 |
+
"source": [
|
492 |
+
"response = travel_agent_executor.invoke({\"input\": 'Eu planejo visitar praias e restaurantes regionais'})\n",
|
493 |
+
"\n",
|
494 |
+
"print(response[\"output\"])"
|
495 |
+
]
|
496 |
+
},
|
497 |
+
{
|
498 |
+
"cell_type": "code",
|
499 |
+
"execution_count": 18,
|
500 |
+
"metadata": {},
|
501 |
+
"outputs": [
|
502 |
+
{
|
503 |
+
"name": "stdout",
|
504 |
+
"output_type": "stream",
|
505 |
+
"text": [
|
506 |
+
"\n",
|
507 |
+
"\n",
|
508 |
+
"\u001b[1m> Entering new AgentExecutor chain...\u001b[0m\n"
|
509 |
+
]
|
510 |
+
},
|
511 |
+
{
|
512 |
+
"name": "stderr",
|
513 |
+
"output_type": "stream",
|
514 |
+
"text": [
|
515 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
516 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
517 |
+
]
|
518 |
+
},
|
519 |
+
{
|
520 |
+
"name": "stdout",
|
521 |
+
"output_type": "stream",
|
522 |
+
"text": [
|
523 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
524 |
+
"Thought: Do I need to use a tool? No\n",
|
525 |
+
"Final Answer: Entendo. Vamos prosseguir sem um orçamento definido por enquanto. Você segue alguma dieta específica ou tem restrições alimentares, como alergias, que devo levar em consideração?\n",
|
526 |
+
"```\n",
|
527 |
+
"\u001b[0m\n",
|
528 |
+
"\n",
|
529 |
+
"\u001b[1m> Finished chain.\u001b[0m\n",
|
530 |
+
"Entendo. Vamos prosseguir sem um orçamento definido por enquanto. Você segue alguma dieta específica ou tem restrições alimentares, como alergias, que devo levar em consideração?\n",
|
531 |
+
"```\n"
|
532 |
+
]
|
533 |
+
}
|
534 |
+
],
|
535 |
+
"source": [
|
536 |
+
"response = travel_agent_executor.invoke({\"input\": 'Eu ainda não sei'})\n",
|
537 |
+
"\n",
|
538 |
+
"print(response[\"output\"])"
|
539 |
+
]
|
540 |
+
},
|
541 |
+
{
|
542 |
+
"cell_type": "code",
|
543 |
+
"execution_count": 19,
|
544 |
+
"metadata": {},
|
545 |
+
"outputs": [
|
546 |
+
{
|
547 |
+
"name": "stdout",
|
548 |
+
"output_type": "stream",
|
549 |
+
"text": [
|
550 |
+
"\n",
|
551 |
+
"\n",
|
552 |
+
"\u001b[1m> Entering new AgentExecutor chain...\u001b[0m\n"
|
553 |
+
]
|
554 |
+
},
|
555 |
+
{
|
556 |
+
"name": "stderr",
|
557 |
+
"output_type": "stream",
|
558 |
+
"text": [
|
559 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
560 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
561 |
+
]
|
562 |
+
},
|
563 |
+
{
|
564 |
+
"name": "stdout",
|
565 |
+
"output_type": "stream",
|
566 |
+
"text": [
|
567 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
568 |
+
"Thought: Do I need to use a tool? No\n",
|
569 |
+
"Final Answer: Ótimo! Sem restrições alimentares. Quantos dias você pretende passar em Natal?\n",
|
570 |
+
"```\n",
|
571 |
+
"\u001b[0m\n",
|
572 |
+
"\n",
|
573 |
+
"\u001b[1m> Finished chain.\u001b[0m\n",
|
574 |
+
"Ótimo! Sem restrições alimentares. Quantos dias você pretende passar em Natal?\n",
|
575 |
+
"```\n"
|
576 |
+
]
|
577 |
+
}
|
578 |
+
],
|
579 |
+
"source": [
|
580 |
+
"response = travel_agent_executor.invoke({\"input\": 'Não possuo nenhuma restrição alimentar.'})\n",
|
581 |
+
"\n",
|
582 |
+
"print(response[\"output\"])"
|
583 |
+
]
|
584 |
+
},
|
585 |
+
{
|
586 |
+
"cell_type": "code",
|
587 |
+
"execution_count": 20,
|
588 |
+
"metadata": {},
|
589 |
+
"outputs": [
|
590 |
+
{
|
591 |
+
"name": "stdout",
|
592 |
+
"output_type": "stream",
|
593 |
+
"text": [
|
594 |
+
"\n",
|
595 |
+
"\n",
|
596 |
+
"\u001b[1m> Entering new AgentExecutor chain...\u001b[0m\n"
|
597 |
+
]
|
598 |
+
},
|
599 |
+
{
|
600 |
+
"name": "stderr",
|
601 |
+
"output_type": "stream",
|
602 |
+
"text": [
|
603 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
604 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
605 |
+
]
|
606 |
+
},
|
607 |
+
{
|
608 |
+
"name": "stdout",
|
609 |
+
"output_type": "stream",
|
610 |
+
"text": [
|
611 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
612 |
+
"Thought: Do I need to use a tool? Yes\n",
|
613 |
+
"Action: Weather Forecast\n",
|
614 |
+
"Action Input: 2025/01/11, 2025/01/12\u001b[0m\u001b[33;1m\u001b[1;3mPrevisão para 2025/01/11, 2025/01/12 em Natal:\n",
|
615 |
+
"\n",
|
616 |
+
"Manhã:\n",
|
617 |
+
"Hora: 05:00 - Temp: 26.3°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 83%\n",
|
618 |
+
"Hora: 06:00 - Temp: 27.6°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 76%\n",
|
619 |
+
"Hora: 07:00 - Temp: 28.5°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 71%\n",
|
620 |
+
"Hora: 08:00 - Temp: 29.5°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 66%\n",
|
621 |
+
"Hora: 09:00 - Temp: 30.3°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 63%\n",
|
622 |
+
"Hora: 10:00 - Temp: 30.6°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 62%\n",
|
623 |
+
"Hora: 11:00 - Temp: 30.6°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 62%\n",
|
624 |
+
"\n",
|
625 |
+
"Tarde:\n",
|
626 |
+
"Hora: 13:00 - Temp: 30.2°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 76%, Umidade: 62%\n",
|
627 |
+
"Hora: 14:00 - Temp: 29.8°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 74%, Umidade: 64%\n",
|
628 |
+
"Hora: 15:00 - Temp: 31.1°C, Condição: Parcialmente nublado, Chance de chuva: 100%, Umidade: 66%\n",
|
629 |
+
"Hora: 16:00 - Temp: 28.1°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 72%\n",
|
630 |
+
"Hora: 17:00 - Temp: 27.2°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 64%, Umidade: 76%\n",
|
631 |
+
"\n",
|
632 |
+
"Noite:\n",
|
633 |
+
"Hora: 18:00 - Temp: 26.9°C, Condição: Parcialmente nublado, Chance de chuva: 0%, Umidade: 76%\n",
|
634 |
+
"Hora: 19:00 - Temp: 26.9°C, Condição: Parcialmente nublado, Chance de chuva: 0%, Umidade: 77%\n",
|
635 |
+
"Hora: 20:00 - Temp: 26.9°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 77%\n",
|
636 |
+
"Hora: 21:00 - Temp: 26.6°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 79%\n",
|
637 |
+
"Hora: 22:00 - Temp: 26.2°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 80%\n",
|
638 |
+
"Hora: 23:00 - Temp: 25.9°C, Condição: Possibilidade de chuva irregular, Chance de chuva: 100%, Umidade: 82%\n",
|
639 |
+
"\u001b[0m"
|
640 |
+
]
|
641 |
+
},
|
642 |
+
{
|
643 |
+
"name": "stderr",
|
644 |
+
"output_type": "stream",
|
645 |
+
"text": [
|
646 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
647 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
648 |
+
]
|
649 |
+
},
|
650 |
+
{
|
651 |
+
"name": "stdout",
|
652 |
+
"output_type": "stream",
|
653 |
+
"text": [
|
654 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
655 |
+
"Thought: Do I need to use a tool? Yes\n",
|
656 |
+
"Action: Query RAG\n",
|
657 |
+
"Action Input: Atrações turísticas em Natal, Rio Grande do Norte, focadas em praias e restaurantes regionais.\u001b[0m\u001b[38;5;200m\u001b[1;3mATRAÇÕES\n",
|
658 |
+
"Em Natal, a vida além da \n",
|
659 |
+
"praia garante aquário, museus, \n",
|
660 |
+
"parque aquático, cajueiro \n",
|
661 |
+
"gigante e estádio de futebol\n",
|
662 |
+
"PARQUE \n",
|
663 |
+
"DAS DUNAS\n",
|
664 |
+
"Um dos grandes \n",
|
665 |
+
"charmes do gigantesco \n",
|
666 |
+
"parque é o Bosque dos \n",
|
667 |
+
"Namorados, que tem \n",
|
668 |
+
"área de piquenique, \n",
|
669 |
+
"playground, restaurante e \n",
|
670 |
+
"centro de visitantes. Vale \n",
|
671 |
+
"a pena também percorrer \n",
|
672 |
+
"as trilhas guiadas do \n",
|
673 |
+
"parque, com vegetação \n",
|
674 |
+
"exuberante. Embora \n",
|
675 |
+
"não tenha praia, oferece \n",
|
676 |
+
"mirantes com lindas vistas \n",
|
677 |
+
"panorâmicas para o mar.\n",
|
678 |
+
"FOTOS: SHUTTERSTOCK\n",
|
679 |
+
"AQUÁRIO NATAL\n",
|
680 |
+
"Na Praia da Redinha, já no \n",
|
681 |
+
"caminho para Genipabu, \n",
|
682 |
+
"possui cerca de 60 \n",
|
683 |
+
"espécies de animais, como \n",
|
684 |
+
"moreias, peixes de corais, \n",
|
685 |
+
"cavalos-marinhos, jacarés e \n",
|
686 |
+
"até mesmo mamíferos, como \n",
|
687 |
+
"hipopótamos e jaguatiricas. \n",
|
688 |
+
"Você pode passar a mão \n",
|
689 |
+
"nas costas dos tubarões e \n",
|
690 |
+
"chegar perto dos pinguins.\n",
|
691 |
+
"\n",
|
692 |
+
"---\n",
|
693 |
+
"\n",
|
694 |
+
"PRAIA A PRAIA\n",
|
695 |
+
"A orla urbana de Natal é marcada por completa estrutura \n",
|
696 |
+
"hoteleira, com quiosques atendendo à beira-mar. Da movimentada \n",
|
697 |
+
"Ponta Negra à tradicional Redinha, tem praia para todos os gostos!\n",
|
698 |
+
"VIA COSTEIRA\n",
|
699 |
+
"A continuação da Praia de Ponta Negra acompanha a famosa Via Costeira, principal \n",
|
700 |
+
"avenida da cidade. Nessa faixa de areia, ficam os incríveis resorts da cidade, com \n",
|
701 |
+
"estrutura completa de lazer e gastronomia. A bela paisagem da região é pontuada por \n",
|
702 |
+
"dunas, trechos de Mata Atlântica e também lindas praias desertas.\n",
|
703 |
+
"PONTA NEGRA\n",
|
704 |
+
"O Morro do Careca é o marco principal da Praia de Ponta Negra e cartão-postal de \n",
|
705 |
+
"Natal. Emoldurada por vegetação, a duna de 107 m de altura domina a paisagem – \n",
|
706 |
+
"mas é proibido subir nela por questões ambientais. A praia em si é supermovimentada,\n",
|
707 |
+
"\n",
|
708 |
+
"---\n",
|
709 |
+
"\n",
|
710 |
+
"PRAIA DA REDINHA\n",
|
711 |
+
"A última praia ao norte de Natal é uma antiga vila de pescadores, cujo destaque é o \n",
|
712 |
+
"Mercado Público. A tradição ali é comer a ginga com tapioca (peixe frito que vem no \n",
|
713 |
+
"palito, dentro de uma tapioca). A Igreja de Nossa Senhora dos Navegantes, construída \n",
|
714 |
+
"com pedras dos arrecifes, e o Aquário Natal são outros pontos de interesse.\n",
|
715 |
+
"PRAIA DO MEIO\n",
|
716 |
+
"Situada entre a Praia do Forte \n",
|
717 |
+
"e a Praia dos Artistas, sedia \n",
|
718 |
+
"anualmente, durante o Réveillon, \n",
|
719 |
+
"a tradicional festa de Iemanjá. Era \n",
|
720 |
+
"considerada a praia mais turística \n",
|
721 |
+
"da cidade até os anos 1980, \n",
|
722 |
+
"devido às formações de piscinas \n",
|
723 |
+
"naturais da região.\n",
|
724 |
+
"PRAIA DOS ARTISTAS\n",
|
725 |
+
"A 5 km do centro de Natal, é uma \n",
|
726 |
+
"das mais agitadas da cidade. Suas \n",
|
727 |
+
"ondas altas são ótimas para surfe \n",
|
728 |
+
"e outros esportes náuticos. Com\n",
|
729 |
+
"\n",
|
730 |
+
"---\n",
|
731 |
+
"\n",
|
732 |
+
"rico de Natal (2010) e Turismo cultural no \n",
|
733 |
+
"centro histórico de Natal (2011).\n",
|
734 |
+
"Em 2012, alguns acontecimentos \n",
|
735 |
+
"vieram a possibilitar a ampliação das \n",
|
736 |
+
"ações: a parceria com o Iphan/RN, no \n",
|
737 |
+
"projeto de extensão “Formação de tu-\n",
|
738 |
+
"tores em educação patrimonial”, a pes-\n",
|
739 |
+
"quisa atrelada às bolsas de iniciação tec-\n",
|
740 |
+
"nológica intitulada “Estudando o centro \n",
|
741 |
+
"histórico de Natal e suas possibilidades \n",
|
742 |
+
"para o turismo”, bem como as atividades \n",
|
743 |
+
"desenvolvidas pelo “Grupo de Pesquisa \n",
|
744 |
+
"Redes Criativas e Colaborativas no Setor \n",
|
745 |
+
"Cultural”.\n",
|
746 |
+
"Entende-se que, apesar de não ser \n",
|
747 |
+
"suficiente, um guia com um roteiro de \n",
|
748 |
+
"visitação é essencial para que o centro \n",
|
749 |
+
"histórico de Natal dê-se a conhecer, e que \n",
|
750 |
+
"este seja um importante passo rumo à \n",
|
751 |
+
"preservação do nosso patrimônio cultu-\n",
|
752 |
+
"ral. Foi construído com a ideia de que,\n",
|
753 |
+
"\n",
|
754 |
+
"---\n",
|
755 |
+
"\n",
|
756 |
+
"EMERENCIANO, João Gothardo Dantas (org.). Natal não – há – \n",
|
757 |
+
"tal: aspectos da história da cidade do Natal. Natal: Departamento \n",
|
758 |
+
"de Informação, Pesquisa e Estatística/SEMURB/Prefeitura Munici-\n",
|
759 |
+
"pal do Natal, 2009.\n",
|
760 |
+
"FERRAZ, Valéria de Souza. Mapeando “esquisitices” no centro \n",
|
761 |
+
"antigo de Natal, RN. In: V SEMINÁRIO DE PESQUISA EM TURIS-\n",
|
762 |
+
"MO DO MERCOSUL – SeminTUR. Turismo: Inovações da Pesquisa \n",
|
763 |
+
"na América Latina. Caxias do Sul, 2008. Anais... Disponível em: \n",
|
764 |
+
"<http://www.usp.br/fau/depprojeto/labcom/produtos/2008_fer-\n",
|
765 |
+
"raz_semintur.pdf>. Acesso em 05 jun. 2012.\n",
|
766 |
+
"__________. Turismo Cultural na ZEPH-Ribeira: possibilidades e \n",
|
767 |
+
"limitações. Dissertação (Mestrado em Arquitetura e Urbanismo). \n",
|
768 |
+
"Universidade Federal do Rio Grande do Norte, Centro de Tecno-\u001b[0m"
|
769 |
+
]
|
770 |
+
},
|
771 |
+
{
|
772 |
+
"name": "stderr",
|
773 |
+
"output_type": "stream",
|
774 |
+
"text": [
|
775 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
776 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
777 |
+
]
|
778 |
+
},
|
779 |
+
{
|
780 |
+
"name": "stdout",
|
781 |
+
"output_type": "stream",
|
782 |
+
"text": [
|
783 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
784 |
+
"Thought: Do I need to use a tool? Yes\n",
|
785 |
+
"Action: DuckDuckGo Search\n",
|
786 |
+
"Action Input: eventos em Natal RN 11 e 12 janeiro 2025\u001b[0m"
|
787 |
+
]
|
788 |
+
},
|
789 |
+
{
|
790 |
+
"name": "stderr",
|
791 |
+
"output_type": "stream",
|
792 |
+
"text": [
|
793 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_community\\utilities\\duckduckgo_search.py:64: UserWarning: 'api' backend is deprecated, using backend='auto'\n",
|
794 |
+
" ddgs_gen = ddgs.text(\n"
|
795 |
+
]
|
796 |
+
},
|
797 |
+
{
|
798 |
+
"name": "stdout",
|
799 |
+
"output_type": "stream",
|
800 |
+
"text": [
|
801 |
+
"\u001b[36;1m\u001b[1;3mA programação cultural de janeiro 2025 em Natal será palco de grandes shows e eventos, garantindo diversão para moradores e turistas. Confira os destaques da programação cultural de janeiro 2025 em Natal e prepare-se para aproveitar o melhor da Cidade do Sol! No dia 11 de janeiro de 2025, a partir das 12h (meio-dia), o Rota 360 , realiza o evento Psy Sunset com várias atrações. Mais informações no perfil oficial @sigarota360. Pura Retrô, Valber Fernandes, Dj PL e Pedro Lucas - Pitanga virou Festa! Ed. Beverly Hills. Picanha Go! Conhece algum local e não está listado? Cadastre aqui! Portal sobre tecnologias, eventos, rally, geoprocessamento e confrarias. O verão de 2025 está em pleno vapor na Grande Natal, trazendo uma variedade de eventos culturais para todos os gostos. As praias de Pirangi se destacam com uma programação intensa, incluindo shows de artistas renomados e festas temáticas que prometem agitar as noites litorâneas. Descubra os melhores eventos em Natal, incluindo shows ao vivo, standups hilariantes, baladas animadas e muito mais. Fique por dentro de toda a diversão e entretenimento que Natal tem a oferecer. Acesse agora e planeje sua próxima experiência inesquecível na cidade!\u001b[0m"
|
802 |
+
]
|
803 |
+
},
|
804 |
+
{
|
805 |
+
"name": "stderr",
|
806 |
+
"output_type": "stream",
|
807 |
+
"text": [
|
808 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
809 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
810 |
+
]
|
811 |
+
},
|
812 |
+
{
|
813 |
+
"name": "stdout",
|
814 |
+
"output_type": "stream",
|
815 |
+
"text": [
|
816 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
817 |
+
"Thought: Do I need to use a tool? No\n",
|
818 |
+
"Final Answer: Ótimo! Com base nas informações que você me forneceu, preparei um roteiro para sua viagem a Natal nos dias 11 e 12 de janeiro de 2025, focando em praias e restaurantes regionais. Lembre-se que a previsão do tempo indica chuva, principalmente no dia 11, então é importante levar roupas apropriadas e estar preparado para possíveis alterações nos planos.\n",
|
819 |
+
"\n",
|
820 |
+
"**Roteiro de Viagem para Natal (11 e 12 de Janeiro de 2025):**\n",
|
821 |
+
"\n",
|
822 |
+
"**Dia 11 de Janeiro (Sábado):**\n",
|
823 |
+
"\n",
|
824 |
+
"* **Manhã (8h-12h):** Considerando a previsão de chuva intensa pela manhã, sugiro começar o dia com um café da manhã reforçado em seu hotel ou em uma cafeteria local. Em seguida, visite o Aquário de Natal na Praia da Redinha. A chuva pode dificultar atividades externas, mas o aquário oferece uma opção coberta e interessante.\n",
|
825 |
+
"* **Tarde (13h-17h):** Se a chuva diminuir, aproveite para visitar a Praia de Ponta Negra e admirar o Morro do Careca (de longe, pois subir é proibido). Caso a chuva persista, você pode explorar o Parque das Dunas, que oferece áreas cobertas e trilhas. Caso a chuva cesse, procure um restaurante regional próximo à Ponta Negra para o almoço.\n",
|
826 |
+
"* **Noite (18h-23h):** A previsão indica chuva novamente à noite. Procure um restaurante com boa comida regional para o jantar, e considere um show ou evento local (verifique a disponibilidade de eventos na data, com base na minha pesquisa no DuckDuckGo, e me diga se deseja que eu procure por algo específico).\n",
|
827 |
+
"\n",
|
828 |
+
"\n",
|
829 |
+
"**Dia 12 de Janeiro (Domingo):**\n",
|
830 |
+
"\n",
|
831 |
+
"* **Manhã (8h-12h):** Comece o dia com um café da manhã tranquilo. Visite a Praia do Meio ou a Praia dos Artistas, dependendo das condições climáticas. Se o tempo estiver bom, aproveite para relaxar na praia. Se chover, visite o Mercado Público da Redinha para experimentar a ginga com tapioca.\n",
|
832 |
+
"* **Tarde (13h-17h):** Explore a Via Costeira, com seus resorts e paisagens bonitas. Escolha um restaurante regional na Via Costeira para o almoço, aproveitando a vista.\n",
|
833 |
+
"* **Noite (18h-23h):** Jante em um restaurante de sua preferência e aproveite a noite em Natal.\n",
|
834 |
+
"\n",
|
835 |
+
"\n",
|
836 |
+
"**Restaurantes Regionais Sugeridos (pesquise mais opções online):**\n",
|
837 |
+
"\n",
|
838 |
+
"* (Pesquise e adicione alguns restaurantes regionais populares em Natal aqui, usando o DuckDuckGo ou outra ferramenta de busca).\n",
|
839 |
+
"\n",
|
840 |
+
"\n",
|
841 |
+
"Este é um roteiro sugestivo, adaptável às condições climáticas e suas preferências. Gostaria de adicionar ou modificar alguma atividade?\n",
|
842 |
+
"\n",
|
843 |
+
"Após finalizarmos, você gostaria que eu salve esses eventos no seu Calendário Google?\n",
|
844 |
+
"```\n",
|
845 |
+
"\u001b[0m\n",
|
846 |
+
"\n",
|
847 |
+
"\u001b[1m> Finished chain.\u001b[0m\n",
|
848 |
+
"Ótimo! Com base nas informações que você me forneceu, preparei um roteiro para sua viagem a Natal nos dias 11 e 12 de janeiro de 2025, focando em praias e restaurantes regionais. Lembre-se que a previsão do tempo indica chuva, principalmente no dia 11, então é importante levar roupas apropriadas e estar preparado para possíveis alterações nos planos.\n",
|
849 |
+
"\n",
|
850 |
+
"**Roteiro de Viagem para Natal (11 e 12 de Janeiro de 2025):**\n",
|
851 |
+
"\n",
|
852 |
+
"**Dia 11 de Janeiro (Sábado):**\n",
|
853 |
+
"\n",
|
854 |
+
"* **Manhã (8h-12h):** Considerando a previsão de chuva intensa pela manhã, sugiro começar o dia com um café da manhã reforçado em seu hotel ou em uma cafeteria local. Em seguida, visite o Aquário de Natal na Praia da Redinha. A chuva pode dificultar atividades externas, mas o aquário oferece uma opção coberta e interessante.\n",
|
855 |
+
"* **Tarde (13h-17h):** Se a chuva diminuir, aproveite para visitar a Praia de Ponta Negra e admirar o Morro do Careca (de longe, pois subir é proibido). Caso a chuva persista, você pode explorar o Parque das Dunas, que oferece áreas cobertas e trilhas. Caso a chuva cesse, procure um restaurante regional próximo à Ponta Negra para o almoço.\n",
|
856 |
+
"* **Noite (18h-23h):** A previsão indica chuva novamente à noite. Procure um restaurante com boa comida regional para o jantar, e considere um show ou evento local (verifique a disponibilidade de eventos na data, com base na minha pesquisa no DuckDuckGo, e me diga se deseja que eu procure por algo específico).\n",
|
857 |
+
"\n",
|
858 |
+
"\n",
|
859 |
+
"**Dia 12 de Janeiro (Domingo):**\n",
|
860 |
+
"\n",
|
861 |
+
"* **Manhã (8h-12h):** Comece o dia com um café da manhã tranquilo. Visite a Praia do Meio ou a Praia dos Artistas, dependendo das condições climáticas. Se o tempo estiver bom, aproveite para relaxar na praia. Se chover, visite o Mercado Público da Redinha para experimentar a ginga com tapioca.\n",
|
862 |
+
"* **Tarde (13h-17h):** Explore a Via Costeira, com seus resorts e paisagens bonitas. Escolha um restaurante regional na Via Costeira para o almoço, aproveitando a vista.\n",
|
863 |
+
"* **Noite (18h-23h):** Jante em um restaurante de sua preferência e aproveite a noite em Natal.\n",
|
864 |
+
"\n",
|
865 |
+
"\n",
|
866 |
+
"**Restaurantes Regionais Sugeridos (pesquise mais opções online):**\n",
|
867 |
+
"\n",
|
868 |
+
"* (Pesquise e adicione alguns restaurantes regionais populares em Natal aqui, usando o DuckDuckGo ou outra ferramenta de busca).\n",
|
869 |
+
"\n",
|
870 |
+
"\n",
|
871 |
+
"Este é um roteiro sugestivo, adaptável às condições climáticas e suas preferências. Gostaria de adicionar ou modificar alguma atividade?\n",
|
872 |
+
"\n",
|
873 |
+
"Após finalizarmos, você gostaria que eu salve esses eventos no seu Calendário Google?\n",
|
874 |
+
"```\n"
|
875 |
+
]
|
876 |
+
}
|
877 |
+
],
|
878 |
+
"source": [
|
879 |
+
"response = travel_agent_executor.invoke({\"input\": 'Eu planejo passar 2 dias da sábado dia 11 de janeiro ao domingo dia 12'})\n",
|
880 |
+
"\n",
|
881 |
+
"print(response[\"output\"])"
|
882 |
+
]
|
883 |
+
},
|
884 |
+
{
|
885 |
+
"cell_type": "code",
|
886 |
+
"execution_count": 21,
|
887 |
+
"metadata": {},
|
888 |
+
"outputs": [
|
889 |
+
{
|
890 |
+
"name": "stdout",
|
891 |
+
"output_type": "stream",
|
892 |
+
"text": [
|
893 |
+
"\n",
|
894 |
+
"\n",
|
895 |
+
"\u001b[1m> Entering new AgentExecutor chain...\u001b[0m\n"
|
896 |
+
]
|
897 |
+
},
|
898 |
+
{
|
899 |
+
"name": "stderr",
|
900 |
+
"output_type": "stream",
|
901 |
+
"text": [
|
902 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
903 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
904 |
+
]
|
905 |
+
},
|
906 |
+
{
|
907 |
+
"name": "stdout",
|
908 |
+
"output_type": "stream",
|
909 |
+
"text": [
|
910 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
911 |
+
"Thought: Do I need to use a tool? Yes\n",
|
912 |
+
"Action: Calendar Agent\n",
|
913 |
+
"Action Input: Salve os seguintes eventos no meu Calendário Google, em um novo calendário chamado \"Janeiro em Natal 2025\":\n",
|
914 |
+
"\n",
|
915 |
+
"**Dia 11 de Janeiro (Sábado):**\n",
|
916 |
+
"\n",
|
917 |
+
"* **Manhã (8h-12h):** Café da manhã e visita ao Aquário de Natal na Praia da Redinha.\n",
|
918 |
+
"* **Tarde (13h-17h):** Praia de Ponta Negra (se o tempo permitir) ou Parque das Dunas (se chover) e almoço em restaurante regional próximo à Ponta Negra.\n",
|
919 |
+
"* **Noite (18h-23h):** Jantar em restaurante regional e possível show ou evento local (a definir).\n",
|
920 |
+
"\n",
|
921 |
+
"\n",
|
922 |
+
"**Dia 12 de Janeiro (Domingo):**\n",
|
923 |
+
"\n",
|
924 |
+
"* **Manhã (8h-12h):** Café da manhã e visita à Praia do Meio ou Praia dos Artistas (dependendo do tempo) ou Mercado Público da Redinha (se chover).\n",
|
925 |
+
"* **Tarde (13h-17h):** Exploração da Via Costeira e almoço em restaurante regional na Via Costeira.\n",
|
926 |
+
"* **Noite (18h-23h):** Jantar em restaurante de preferência.\n",
|
927 |
+
"\n",
|
928 |
+
"Por favor, adicione detalhes sobre os restaurantes assim que eu os pesquisar e te informar.\u001b[0m\n",
|
929 |
+
"\n",
|
930 |
+
"\u001b[1m> Entering new AgentExecutor chain...\u001b[0m\n"
|
931 |
+
]
|
932 |
+
},
|
933 |
+
{
|
934 |
+
"name": "stderr",
|
935 |
+
"output_type": "stream",
|
936 |
+
"text": [
|
937 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
938 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
939 |
+
]
|
940 |
+
},
|
941 |
+
{
|
942 |
+
"name": "stdout",
|
943 |
+
"output_type": "stream",
|
944 |
+
"text": [
|
945 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
946 |
+
"Thought: Do I need to use a tool? Yes\n",
|
947 |
+
"Action: Create Calendar List\n",
|
948 |
+
"Action Input: Janeiro em Natal 2025\u001b[0m\u001b[36;1m\u001b[1;3m{'kind': 'calendar#calendar', 'etag': '\"EitsE6nkIJY5B1bHCZi9_eFLkxo\"', 'id': '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com', 'summary': 'Janeiro em Natal 2025', 'timeZone': 'UTC', 'conferenceProperties': {'allowedConferenceSolutionTypes': ['hangoutsMeet']}}\u001b[0m"
|
949 |
+
]
|
950 |
+
},
|
951 |
+
{
|
952 |
+
"name": "stderr",
|
953 |
+
"output_type": "stream",
|
954 |
+
"text": [
|
955 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
956 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
957 |
+
]
|
958 |
+
},
|
959 |
+
{
|
960 |
+
"name": "stdout",
|
961 |
+
"output_type": "stream",
|
962 |
+
"text": [
|
963 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
964 |
+
"Thought: Do I need to use a tool? Yes\n",
|
965 |
+
"Action: Insert Calendar Event\n",
|
966 |
+
"Action Input: Agendar evento no calendário '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com': Café da manhã e visita ao Aquário de Natal na Praia da Redinha. Início: 11/01/2025 08:00, Término: 11/01/2025 12:00, Local: Aquário de Natal, Praia da Redinha.\u001b[0m"
|
967 |
+
]
|
968 |
+
},
|
969 |
+
{
|
970 |
+
"name": "stderr",
|
971 |
+
"output_type": "stream",
|
972 |
+
"text": [
|
973 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
974 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
975 |
+
]
|
976 |
+
},
|
977 |
+
{
|
978 |
+
"name": "stdout",
|
979 |
+
"output_type": "stream",
|
980 |
+
"text": [
|
981 |
+
"\u001b[36;1m\u001b[1;3m{'kind': 'calendar#event', 'etag': '\"3473241538680000\"', 'id': 'hcepfmnmbhcc3lji74raci603c', 'status': 'confirmed', 'htmlLink': 'https://www.google.com/calendar/event?eid=aGNlcGZtbm1iaGNjM2xqaTc0cmFjaTYwM2MgNGU2YTI1MWI5MTIzOTYzODcwNTQzMTFiMDRhMzgzZTU2ODhjNTJmOTQ2ZjhkNjQwMTQzZTg2OWNkYzUyOTA3OEBn', 'created': '2025-01-11T18:39:29.000Z', 'updated': '2025-01-11T18:39:29.340Z', 'summary': 'Café da manhã e visita ao Aquário de Natal', 'description': 'Café da manhã e visita ao Aquário de Natal na Praia da Redinha.', 'location': 'Aquário de Natal, Praia da Redinha', 'creator': {'email': '[email protected]'}, 'organizer': {'email': '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com', 'displayName': 'Janeiro em Natal 2025', 'self': True}, 'start': {'dateTime': '2025-01-11T11:00:00Z', 'timeZone': 'America/Fortaleza'}, 'end': {'dateTime': '2025-01-11T15:00:00Z', 'timeZone': 'America/Fortaleza'}, 'iCalUID': '[email protected]', 'sequence': 0, 'reminders': {'useDefault': True}, 'eventType': 'default'}\u001b[0m"
|
982 |
+
]
|
983 |
+
},
|
984 |
+
{
|
985 |
+
"name": "stderr",
|
986 |
+
"output_type": "stream",
|
987 |
+
"text": [
|
988 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
989 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
990 |
+
]
|
991 |
+
},
|
992 |
+
{
|
993 |
+
"name": "stdout",
|
994 |
+
"output_type": "stream",
|
995 |
+
"text": [
|
996 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
997 |
+
"Thought: Do I need to use a tool? Yes\n",
|
998 |
+
"Action: Insert Calendar Event\n",
|
999 |
+
"Action Input: Agendar evento no calendário '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com': Praia de Ponta Negra (se o tempo permitir) ou Parque das Dunas (se chover) e almoço em restaurante regional próximo à Ponta Negra. Início: 11/01/2025 13:00, Término: 11/01/2025 17:00, Local: Ponta Negra/Parque das Dunas, Natal.\u001b[0m"
|
1000 |
+
]
|
1001 |
+
},
|
1002 |
+
{
|
1003 |
+
"name": "stderr",
|
1004 |
+
"output_type": "stream",
|
1005 |
+
"text": [
|
1006 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1007 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1008 |
+
]
|
1009 |
+
},
|
1010 |
+
{
|
1011 |
+
"name": "stdout",
|
1012 |
+
"output_type": "stream",
|
1013 |
+
"text": [
|
1014 |
+
"\u001b[36;1m\u001b[1;3m{'kind': 'calendar#event', 'etag': '\"3473241550376000\"', 'id': 'qslpqd5tpbsjnmmadugs4bgtfk', 'status': 'confirmed', 'htmlLink': 'https://www.google.com/calendar/event?eid=cXNscHFkNXRwYnNqbm1tYWR1Z3M0Ymd0ZmsgNGU2YTI1MWI5MTIzOTYzODcwNTQzMTFiMDRhMzgzZTU2ODhjNTJmOTQ2ZjhkNjQwMTQzZTg2OWNkYzUyOTA3OEBn', 'created': '2025-01-11T18:39:35.000Z', 'updated': '2025-01-11T18:39:35.188Z', 'summary': 'Praia de Ponta Negra ou Parque das Dunas e almoço', 'description': 'Praia de Ponta Negra (se o tempo permitir) ou Parque das Dunas (se chover) e almoço em restaurante regional próximo à Ponta Negra.', 'location': 'Ponta Negra/Parque das Dunas, Natal', 'creator': {'email': '[email protected]'}, 'organizer': {'email': '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com', 'displayName': 'Janeiro em Natal 2025', 'self': True}, 'start': {'dateTime': '2025-01-11T16:00:00Z', 'timeZone': 'America/Fortaleza'}, 'end': {'dateTime': '2025-01-11T20:00:00Z', 'timeZone': 'America/Fortaleza'}, 'iCalUID': '[email protected]', 'sequence': 0, 'reminders': {'useDefault': True}, 'eventType': 'default'}\u001b[0m"
|
1015 |
+
]
|
1016 |
+
},
|
1017 |
+
{
|
1018 |
+
"name": "stderr",
|
1019 |
+
"output_type": "stream",
|
1020 |
+
"text": [
|
1021 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1022 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1023 |
+
]
|
1024 |
+
},
|
1025 |
+
{
|
1026 |
+
"name": "stdout",
|
1027 |
+
"output_type": "stream",
|
1028 |
+
"text": [
|
1029 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
1030 |
+
"Thought: Do I need to use a tool? Yes\n",
|
1031 |
+
"Action: Insert Calendar Event\n",
|
1032 |
+
"Action Input: Agendar evento no calendário '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com': Jantar em restaurante regional e possível show ou evento local (a definir). Início: 11/01/2025 18:00, Término: 11/01/2025 23:00, Local: A definir, Natal.\u001b[0m"
|
1033 |
+
]
|
1034 |
+
},
|
1035 |
+
{
|
1036 |
+
"name": "stderr",
|
1037 |
+
"output_type": "stream",
|
1038 |
+
"text": [
|
1039 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1040 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1041 |
+
]
|
1042 |
+
},
|
1043 |
+
{
|
1044 |
+
"name": "stdout",
|
1045 |
+
"output_type": "stream",
|
1046 |
+
"text": [
|
1047 |
+
"\u001b[36;1m\u001b[1;3m{'kind': 'calendar#event', 'etag': '\"3473241562330000\"', 'id': 'k5n5tvdn8n2fvtkk6loupv6snk', 'status': 'confirmed', 'htmlLink': 'https://www.google.com/calendar/event?eid=azVuNXR2ZG44bjJmdnRrazZsb3VwdjZzbmsgNGU2YTI1MWI5MTIzOTYzODcwNTQzMTFiMDRhMzgzZTU2ODhjNTJmOTQ2ZjhkNjQwMTQzZTg2OWNkYzUyOTA3OEBn', 'created': '2025-01-11T18:39:41.000Z', 'updated': '2025-01-11T18:39:41.165Z', 'summary': 'Jantar em restaurante regional e possível show ou evento local (a definir)', 'description': 'Jantar em restaurante regional e possível show ou evento local (a definir).', 'location': 'A definir, Natal', 'creator': {'email': '[email protected]'}, 'organizer': {'email': '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com', 'displayName': 'Janeiro em Natal 2025', 'self': True}, 'start': {'dateTime': '2025-01-11T21:00:00Z', 'timeZone': 'America/Fortaleza'}, 'end': {'dateTime': '2025-01-12T02:00:00Z', 'timeZone': 'America/Fortaleza'}, 'iCalUID': '[email protected]', 'sequence': 0, 'reminders': {'useDefault': True}, 'eventType': 'default'}\u001b[0m"
|
1048 |
+
]
|
1049 |
+
},
|
1050 |
+
{
|
1051 |
+
"name": "stderr",
|
1052 |
+
"output_type": "stream",
|
1053 |
+
"text": [
|
1054 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1055 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1056 |
+
]
|
1057 |
+
},
|
1058 |
+
{
|
1059 |
+
"name": "stdout",
|
1060 |
+
"output_type": "stream",
|
1061 |
+
"text": [
|
1062 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
1063 |
+
"Thought: Do I need to use a tool? Yes\n",
|
1064 |
+
"Action: Insert Calendar Event\n",
|
1065 |
+
"Action Input: Agendar evento no calendário '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com': Café da manhã e visita à Praia do Meio ou Praia dos Artistas (dependendo do tempo) ou Mercado Público da Redinha (se chover). Início: 12/01/2025 08:00, Término: 12/01/2025 12:00, Local: Praia do Meio/Praia dos Artistas/Mercado Público da Redinha, Natal.\u001b[0m"
|
1066 |
+
]
|
1067 |
+
},
|
1068 |
+
{
|
1069 |
+
"name": "stderr",
|
1070 |
+
"output_type": "stream",
|
1071 |
+
"text": [
|
1072 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1073 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1074 |
+
]
|
1075 |
+
},
|
1076 |
+
{
|
1077 |
+
"name": "stdout",
|
1078 |
+
"output_type": "stream",
|
1079 |
+
"text": [
|
1080 |
+
"\u001b[36;1m\u001b[1;3m{'kind': 'calendar#event', 'etag': '\"3473241575456000\"', 'id': 'm5tkc64g7e4o25la20kr46j65o', 'status': 'confirmed', 'htmlLink': 'https://www.google.com/calendar/event?eid=bTV0a2M2NGc3ZTRvMjVsYTIwa3I0Nmo2NW8gNGU2YTI1MWI5MTIzOTYzODcwNTQzMTFiMDRhMzgzZTU2ODhjNTJmOTQ2ZjhkNjQwMTQzZTg2OWNkYzUyOTA3OEBn', 'created': '2025-01-11T18:39:47.000Z', 'updated': '2025-01-11T18:39:47.728Z', 'summary': 'Café da manhã e visita à praia', 'description': 'Café da manhã e visita à Praia do Meio ou Praia dos Artistas (dependendo do tempo) ou Mercado Público da Redinha (se chover).', 'location': 'Praia do Meio/Praia dos Artistas/Mercado Público da Redinha, Natal', 'creator': {'email': '[email protected]'}, 'organizer': {'email': '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com', 'displayName': 'Janeiro em Natal 2025', 'self': True}, 'start': {'dateTime': '2025-01-12T11:00:00Z', 'timeZone': 'America/Fortaleza'}, 'end': {'dateTime': '2025-01-12T15:00:00Z', 'timeZone': 'America/Fortaleza'}, 'iCalUID': '[email protected]', 'sequence': 0, 'reminders': {'useDefault': True}, 'eventType': 'default'}\u001b[0m"
|
1081 |
+
]
|
1082 |
+
},
|
1083 |
+
{
|
1084 |
+
"name": "stderr",
|
1085 |
+
"output_type": "stream",
|
1086 |
+
"text": [
|
1087 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1088 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1089 |
+
]
|
1090 |
+
},
|
1091 |
+
{
|
1092 |
+
"name": "stdout",
|
1093 |
+
"output_type": "stream",
|
1094 |
+
"text": [
|
1095 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
1096 |
+
"Thought: Do I need to use a tool? Yes\n",
|
1097 |
+
"Action: Insert Calendar Event\n",
|
1098 |
+
"Action Input: Agendar evento no calendário '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com': Exploração da Via Costeira e almoço em restaurante regional na Via Costeira. Início: 12/01/2025 13:00, Término: 12/01/2025 17:00, Local: Via Costeira, Natal.\u001b[0m"
|
1099 |
+
]
|
1100 |
+
},
|
1101 |
+
{
|
1102 |
+
"name": "stderr",
|
1103 |
+
"output_type": "stream",
|
1104 |
+
"text": [
|
1105 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1106 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1107 |
+
]
|
1108 |
+
},
|
1109 |
+
{
|
1110 |
+
"name": "stdout",
|
1111 |
+
"output_type": "stream",
|
1112 |
+
"text": [
|
1113 |
+
"\u001b[36;1m\u001b[1;3m{'kind': 'calendar#event', 'etag': '\"3473241586958000\"', 'id': '3m0gqf50fkmsalagq11q0d83eg', 'status': 'confirmed', 'htmlLink': 'https://www.google.com/calendar/event?eid=M20wZ3FmNTBma21zYWxhZ3ExMXEwZDgzZWcgNGU2YTI1MWI5MTIzOTYzODcwNTQzMTFiMDRhMzgzZTU2ODhjNTJmOTQ2ZjhkNjQwMTQzZTg2OWNkYzUyOTA3OEBn', 'created': '2025-01-11T18:39:53.000Z', 'updated': '2025-01-11T18:39:53.479Z', 'summary': 'Exploração da Via Costeira e almoço em restaurante regional na Via Costeira', 'location': 'Via Costeira, Natal', 'creator': {'email': '[email protected]'}, 'organizer': {'email': '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com', 'displayName': 'Janeiro em Natal 2025', 'self': True}, 'start': {'dateTime': '2025-01-12T16:00:00Z', 'timeZone': 'America/Fortaleza'}, 'end': {'dateTime': '2025-01-12T20:00:00Z', 'timeZone': 'America/Fortaleza'}, 'iCalUID': '[email protected]', 'sequence': 0, 'reminders': {'useDefault': True}, 'eventType': 'default'}\u001b[0m"
|
1114 |
+
]
|
1115 |
+
},
|
1116 |
+
{
|
1117 |
+
"name": "stderr",
|
1118 |
+
"output_type": "stream",
|
1119 |
+
"text": [
|
1120 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1121 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1122 |
+
]
|
1123 |
+
},
|
1124 |
+
{
|
1125 |
+
"name": "stdout",
|
1126 |
+
"output_type": "stream",
|
1127 |
+
"text": [
|
1128 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
1129 |
+
"Thought: Do I need to use a tool? Yes\n",
|
1130 |
+
"Action: Insert Calendar Event\n",
|
1131 |
+
"Action Input: Agendar evento no calendário '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com': Jantar em restaurante de preferência. Início: 12/01/2025 18:00, Término: 12/01/2025 23:00, Local: A definir, Natal.\u001b[0m"
|
1132 |
+
]
|
1133 |
+
},
|
1134 |
+
{
|
1135 |
+
"name": "stderr",
|
1136 |
+
"output_type": "stream",
|
1137 |
+
"text": [
|
1138 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1139 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1140 |
+
]
|
1141 |
+
},
|
1142 |
+
{
|
1143 |
+
"name": "stdout",
|
1144 |
+
"output_type": "stream",
|
1145 |
+
"text": [
|
1146 |
+
"\u001b[36;1m\u001b[1;3m{'kind': 'calendar#event', 'etag': '\"3473241598586000\"', 'id': 'ane3ojkf8am2pu6tpkmln5u604', 'status': 'confirmed', 'htmlLink': 'https://www.google.com/calendar/event?eid=YW5lM29qa2Y4YW0ycHU2dHBrbWxuNXU2MDQgNGU2YTI1MWI5MTIzOTYzODcwNTQzMTFiMDRhMzgzZTU2ODhjNTJmOTQ2ZjhkNjQwMTQzZTg2OWNkYzUyOTA3OEBn', 'created': '2025-01-11T18:39:59.000Z', 'updated': '2025-01-11T18:39:59.293Z', 'summary': 'Jantar em restaurante de preferência', 'location': 'A definir, Natal', 'creator': {'email': '[email protected]'}, 'organizer': {'email': '4e6a251b912396387054311b04a383e5688c52f946f8d640143e869cdc529078@group.calendar.google.com', 'displayName': 'Janeiro em Natal 2025', 'self': True}, 'start': {'dateTime': '2025-01-12T21:00:00Z', 'timeZone': 'America/Fortaleza'}, 'end': {'dateTime': '2025-01-13T02:00:00Z', 'timeZone': 'America/Fortaleza'}, 'iCalUID': '[email protected]', 'sequence': 0, 'reminders': {'useDefault': True}, 'eventType': 'default'}\u001b[0m"
|
1147 |
+
]
|
1148 |
+
},
|
1149 |
+
{
|
1150 |
+
"name": "stderr",
|
1151 |
+
"output_type": "stream",
|
1152 |
+
"text": [
|
1153 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1154 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1155 |
+
]
|
1156 |
+
},
|
1157 |
+
{
|
1158 |
+
"name": "stdout",
|
1159 |
+
"output_type": "stream",
|
1160 |
+
"text": [
|
1161 |
+
"\u001b[32;1m\u001b[1;3m```tool_code\n",
|
1162 |
+
"Thought: Do I need to use a tool? No\n",
|
1163 |
+
"Final Answer: Ótimo! Agendei todos os eventos no seu novo calendário \"Janeiro em Natal 2025\". Lembre-se que alguns detalhes, como os nomes dos restaurantes, ainda precisam ser adicionados após você pesquisar e me informar. Assim que tiver essas informações, posso atualizar o calendário com os detalhes completos. Você pode acessar seu calendário Google para visualizar os eventos agendados. Precisará de mais alguma coisa?\n",
|
1164 |
+
"```\n",
|
1165 |
+
"\u001b[0m\n",
|
1166 |
+
"\n",
|
1167 |
+
"\u001b[1m> Finished chain.\u001b[0m\n",
|
1168 |
+
"\u001b[36;1m\u001b[1;3m{'input': 'Salve os seguintes eventos no meu Calendário Google, em um novo calendário chamado \"Janeiro em Natal 2025\":\\n\\n**Dia 11 de Janeiro (Sábado):**\\n\\n* **Manhã (8h-12h):** Café da manhã e visita ao Aquário de Natal na Praia da Redinha.\\n* **Tarde (13h-17h):** Praia de Ponta Negra (se o tempo permitir) ou Parque das Dunas (se chover) e almoço em restaurante regional próximo à Ponta Negra.\\n* **Noite (18h-23h):** Jantar em restaurante regional e possível show ou evento local (a definir).\\n\\n\\n**Dia 12 de Janeiro (Domingo):**\\n\\n* **Manhã (8h-12h):** Café da manhã e visita à Praia do Meio ou Praia dos Artistas (dependendo do tempo) ou Mercado Público da Redinha (se chover).\\n* **Tarde (13h-17h):** Exploração da Via Costeira e almoço em restaurante regional na Via Costeira.\\n* **Noite (18h-23h):** Jantar em restaurante de preferência.\\n\\nPor favor, adicione detalhes sobre os restaurantes assim que eu os pesquisar e te informar.', 'chat_history': 'Human: Boa noite\\nAI: Boa noite! Para começarmos a planejar sua viagem, preciso de algumas informações. Para onde você está planejando viajar?\\nHuman: Eu estou planejando visitar Natal no Rio Grande do Norte\\nAI: Ótimo! Natal é uma excelente escolha. Quais atividades ou experiências você está buscando nessa viagem? Por exemplo, prefere explorar praias, restaurantes, eventos, museus, história, cultura, vida noturna, natureza, ou algo mais?\\nHuman: Eu planejo visitar praias e restaurantes regionais\\nAI: Perfeito! Praias e restaurantes regionais em Natal. Tem um orçamento em mente ou um limite de gastos que gostaria de seguir durante a viagem?\\n```\\nHuman: Eu ainda não sei\\nAI: Entendo. Vamos prosseguir sem um orçamento definido por enquanto. Você segue alguma dieta específica ou tem restrições alimentares, como alergias, que devo levar em consideração?\\n```\\nHuman: Não possuo nenhuma restrição alimentar.\\nAI: Ótimo! Sem restrições alimentares. Quantos dias você pretende passar em Natal?\\n```\\nHuman: Eu planejo passar 2 dias da sábado dia 11 de janeiro ao domingo dia 12\\nAI: Ótimo! Com base nas informações que você me forneceu, preparei um roteiro para sua viagem a Natal nos dias 11 e 12 de janeiro de 2025, focando em praias e restaurantes regionais. Lembre-se que a previsão do tempo indica chuva, principalmente no dia 11, então é importante levar roupas apropriadas e estar preparado para possíveis alterações nos planos.\\n\\n**Roteiro de Viagem para Natal (11 e 12 de Janeiro de 2025):**\\n\\n**Dia 11 de Janeiro (Sábado):**\\n\\n* **Manhã (8h-12h):** Considerando a previsão de chuva intensa pela manhã, sugiro começar o dia com um café da manhã reforçado em seu hotel ou em uma cafeteria local. Em seguida, visite o Aquário de Natal na Praia da Redinha. A chuva pode dificultar atividades externas, mas o aquário oferece uma opção coberta e interessante.\\n* **Tarde (13h-17h):** Se a chuva diminuir, aproveite para visitar a Praia de Ponta Negra e admirar o Morro do Careca (de longe, pois subir é proibido). Caso a chuva persista, você pode explorar o Parque das Dunas, que oferece áreas cobertas e trilhas. Caso a chuva cesse, procure um restaurante regional próximo à Ponta Negra para o almoço.\\n* **Noite (18h-23h):** A previsão indica chuva novamente à noite. Procure um restaurante com boa comida regional para o jantar, e considere um show ou evento local (verifique a disponibilidade de eventos na data, com base na minha pesquisa no DuckDuckGo, e me diga se deseja que eu procure por algo específico).\\n\\n\\n**Dia 12 de Janeiro (Domingo):**\\n\\n* **Manhã (8h-12h):** Comece o dia com um café da manhã tranquilo. Visite a Praia do Meio ou a Praia dos Artistas, dependendo das condições climáticas. Se o tempo estiver bom, aproveite para relaxar na praia. Se chover, visite o Mercado Público da Redinha para experimentar a ginga com tapioca.\\n* **Tarde (13h-17h):** Explore a Via Costeira, com seus resorts e paisagens bonitas. Escolha um restaurante regional na Via Costeira para o almoço, aproveitando a vista.\\n* **Noite (18h-23h):** Jante em um restaurante de sua preferência e aproveite a noite em Natal.\\n\\n\\n**Restaurantes Regionais Sugeridos (pesquise mais opções online):**\\n\\n* (Pesquise e adicione alguns restaurantes regionais populares em Natal aqui, usando o DuckDuckGo ou outra ferramenta de busca).\\n\\n\\nEste é um roteiro sugestivo, adaptável às condições climáticas e suas preferências. Gostaria de adicionar ou modificar alguma atividade?\\n\\nApós finalizarmos, você gostaria que eu salve esses eventos no seu Calendário Google?\\n```', 'output': 'Ótimo! Agendei todos os eventos no seu novo calendário \"Janeiro em Natal 2025\". Lembre-se que alguns detalhes, como os nomes dos restaurantes, ainda precisam ser adicionados após você pesquisar e me informar. Assim que tiver essas informações, posso atualizar o calendário com os detalhes completos. Você pode acessar seu calendário Google para visualizar os eventos agendados. Precisará de mais alguma coisa?\\n```'}\u001b[0m"
|
1169 |
+
]
|
1170 |
+
},
|
1171 |
+
{
|
1172 |
+
"name": "stderr",
|
1173 |
+
"output_type": "stream",
|
1174 |
+
"text": [
|
1175 |
+
"c:\\Users\\Tales\\Python\\AgenteTuristico\\.venv\\lib\\site-packages\\langchain_google_genai\\chat_models.py:310: UserWarning: Convert_system_message_to_human will be deprecated!\n",
|
1176 |
+
" warnings.warn(\"Convert_system_message_to_human will be deprecated!\")\n"
|
1177 |
+
]
|
1178 |
+
},
|
1179 |
+
{
|
1180 |
+
"name": "stdout",
|
1181 |
+
"output_type": "stream",
|
1182 |
+
"text": [
|
1183 |
+
"\u001b[32;1m\u001b[1;3mDo I need to use a tool? No\n",
|
1184 |
+
"Final Answer: Ótimo! Agendei todos os eventos no seu novo calendário \"Janeiro em Natal 2025\". Lembre-se que alguns detalhes, como os nomes dos restaurantes, ainda precisam ser adicionados após você pesquisar e me informar. Assim que tiver essas informações, posso atualizar o calendário com os detalhes completos. Você pode acessar seu calendário Google para visualizar os eventos agendados. Precisará de mais alguma coisa?\n",
|
1185 |
+
"\u001b[0m\n",
|
1186 |
+
"\n",
|
1187 |
+
"\u001b[1m> Finished chain.\u001b[0m\n",
|
1188 |
+
"Ótimo! Agendei todos os eventos no seu novo calendário \"Janeiro em Natal 2025\". Lembre-se que alguns detalhes, como os nomes dos restaurantes, ainda precisam ser adicionados após você pesquisar e me informar. Assim que tiver essas informações, posso atualizar o calendário com os detalhes completos. Você pode acessar seu calendário Google para visualizar os eventos agendados. Precisará de mais alguma coisa?\n"
|
1189 |
+
]
|
1190 |
+
}
|
1191 |
+
],
|
1192 |
+
"source": [
|
1193 |
+
"response = travel_agent_executor.invoke({\"input\": 'Eu quero que você salve os eventos sugeridos no meu calendário google, no novo calendário chamado Janeiro em Natal 2025.'})\n",
|
1194 |
+
"\n",
|
1195 |
+
"print(response[\"output\"])"
|
1196 |
+
]
|
1197 |
+
},
|
1198 |
+
{
|
1199 |
+
"cell_type": "code",
|
1200 |
+
"execution_count": null,
|
1201 |
+
"metadata": {},
|
1202 |
+
"outputs": [],
|
1203 |
+
"source": [
|
1204 |
+
"print(response[\"chat_history\"])"
|
1205 |
+
]
|
1206 |
+
}
|
1207 |
+
],
|
1208 |
+
"metadata": {
|
1209 |
+
"colab": {
|
1210 |
+
"provenance": []
|
1211 |
+
},
|
1212 |
+
"kernelspec": {
|
1213 |
+
"display_name": ".venv",
|
1214 |
+
"language": "python",
|
1215 |
+
"name": "python3"
|
1216 |
+
},
|
1217 |
+
"language_info": {
|
1218 |
+
"codemirror_mode": {
|
1219 |
+
"name": "ipython",
|
1220 |
+
"version": 3
|
1221 |
+
},
|
1222 |
+
"file_extension": ".py",
|
1223 |
+
"mimetype": "text/x-python",
|
1224 |
+
"name": "python",
|
1225 |
+
"nbconvert_exporter": "python",
|
1226 |
+
"pygments_lexer": "ipython3",
|
1227 |
+
"version": "3.10.11"
|
1228 |
+
}
|
1229 |
+
},
|
1230 |
+
"nbformat": 4,
|
1231 |
+
"nbformat_minor": 0
|
1232 |
+
}
|
README.md
CHANGED
@@ -1,12 +1,91 @@
|
|
1 |
-
|
2 |
-
|
3 |
-
|
4 |
-
|
5 |
-
|
6 |
-
|
7 |
-
|
8 |
-
|
9 |
-
|
10 |
-
|
11 |
-
|
12 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Agente de Planejamento de Viagens para Natal - RN
|
2 |
+
|
3 |
+
## Descrição do Projeto
|
4 |
+
|
5 |
+
Natal, capital do Rio Grande do Norte, é um destino turístico muito procurado no Brasil, famoso por suas belas praias e cultura nordestina. Muitos turistas que desejam explorar a região preferem planejar suas próprias viagens para economizar dinheiro, em vez de contratar guias turísticos.
|
6 |
+
|
7 |
+
Este projeto tem como objetivo auxiliar esses viajantes independentes, oferecendo um guia turístico personalizado através de uma interface conversacional baseada em inteligência artificial. O sistema utiliza um conjunto de agentes para interagir com os usuários em linguagem natural e, com base nas preferências e informações fornecidas, gerar um roteiro de viagem detalhado, incluindo atividades e sugestões de locais.
|
8 |
+
|
9 |
+
O diferencial deste projeto é a capacidade de criar um planejamento personalizado e, opcionalmente, agendar automaticamente os eventos da viagem no Google Calendar do usuário.
|
10 |
+
|
11 |
+
## Funcionalidades Principais
|
12 |
+
|
13 |
+
* **Interface Conversacional:** Permite aos usuários interagir com o sistema utilizando linguagem natural para planejar suas viagens.
|
14 |
+
* **Agente de Planejamento:** Cria roteiros de viagem personalizados com base nas preferências e interesses do usuário.
|
15 |
+
* **Sugestões Detalhadas:** Inclui sugestões de atividades, atrações turísticas, restaurantes e horários para cada dia da viagem.
|
16 |
+
* **Previsão do Tempo:** Integra a previsão do tempo para os dias da viagem, auxiliando no planejamento.
|
17 |
+
* **Agendamento no Google Calendar (Opcional):** Permite que os usuários agendem automaticamente as atividades do roteiro no Google Calendar.
|
18 |
+
* **Adaptação Personalizada:** Gera roteiros que se adaptam ao perfil e preferências do usuário, incluindo sugestões de acordo com o orçamento e gostos.
|
19 |
+
* **Busca de Informações:** Utiliza ferramentas de busca para obter informações atualizadas sobre atrações, restaurantes e clima.
|
20 |
+
|
21 |
+
## Tecnologias e Ferramentas Utilizadas
|
22 |
+
|
23 |
+
Este projeto utiliza diversas ferramentas de Processamento de Linguagem Natural (PLN) e modelos de linguagem (LLM), juntamente com outras tecnologias essenciais para o funcionamento do sistema:
|
24 |
+
|
25 |
+
* **Modelos de Linguagem (LLM):**
|
26 |
+
* `gemini-1.5-flash` (Google): Modelo para processamento de linguagem natural e geração de respostas.
|
27 |
+
* **Framework de Agentes:**
|
28 |
+
* `Langchain`: Framework para criação de agentes e pipelines de processamento de linguagem natural.
|
29 |
+
* **Ferramentas de Busca e Informação:**
|
30 |
+
* `Retrieval-Augmented Generation (RAG)`: Sistema de busca e recuperação de informações baseado em banco de dados vetoriais.
|
31 |
+
* `DuckDuckGo Search API`: Para busca de informações atualizadas na internet.
|
32 |
+
* `WeatherAPI`: Para obter informações meteorológicas detalhadas de uma determinada data.
|
33 |
+
* **Bancos de Dados Vetoriais:**
|
34 |
+
* `Chroma`: Banco de dados vetorial para armazenamento de documentos e busca semântica.
|
35 |
+
|
36 |
+
### Pré-requisitos
|
37 |
+
|
38 |
+
Certifique-se de ter o Python 3.10 instalado em seu sistema. Você pode verificar a versão do Python instalada usando o seguinte comando:
|
39 |
+
|
40 |
+
```bash
|
41 |
+
python --version
|
42 |
+
```
|
43 |
+
|
44 |
+
### Passo a Passo
|
45 |
+
|
46 |
+
1. **Criação de um Ambiente Virtual:**
|
47 |
+
|
48 |
+
Crie um ambiente virtual utilizando a versão 3.10 do Python:
|
49 |
+
|
50 |
+
```bash
|
51 |
+
python -m venv venv
|
52 |
+
```
|
53 |
+
|
54 |
+
2. **Ativação do Ambiente Virtual:**
|
55 |
+
|
56 |
+
Ative o ambiente virtual criado:
|
57 |
+
|
58 |
+
- No Windows:
|
59 |
+
```bash
|
60 |
+
.\venv\Scripts\activate
|
61 |
+
```
|
62 |
+
- No macOS/Linux:
|
63 |
+
```bash
|
64 |
+
source venv/bin/activate
|
65 |
+
```
|
66 |
+
|
67 |
+
3. **Instalação das Dependências:**
|
68 |
+
|
69 |
+
Com o ambiente virtual ativado, instale as dependências listadas no arquivo `requirements.txt`:
|
70 |
+
|
71 |
+
```bash
|
72 |
+
pip install -r requirements.txt
|
73 |
+
```
|
74 |
+
|
75 |
+
4. **Execução do Script `video_transcriptor.py`:**
|
76 |
+
|
77 |
+
Execute o script `video_transcriptor.py` para transcrever vídeos:
|
78 |
+
|
79 |
+
```bash
|
80 |
+
python video_transcriptor.py
|
81 |
+
```
|
82 |
+
|
83 |
+
5. **Execução do Script `populate_database.py`:**
|
84 |
+
|
85 |
+
Execute o script `populate_database.py` para popular o banco de dados:
|
86 |
+
|
87 |
+
```bash
|
88 |
+
python populate_database.py
|
89 |
+
```
|
90 |
+
|
91 |
+
Pronto! Agora você está pronto para utilizar o sistema de planejamento de viagens.
|
agents.py
ADDED
@@ -0,0 +1,253 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
from datetime import datetime
|
3 |
+
from langchain import hub
|
4 |
+
from langchain.agents import Tool, AgentExecutor
|
5 |
+
from langchain.agents.format_scratchpad import format_log_to_str
|
6 |
+
from langchain.agents.output_parsers import ReActSingleInputOutputParser
|
7 |
+
from langchain.memory import ChatMessageHistory, ConversationBufferWindowMemory
|
8 |
+
from langchain.tools.render import render_text_description
|
9 |
+
from langchain.utilities import DuckDuckGoSearchAPIWrapper
|
10 |
+
from langchain_google_genai import (
|
11 |
+
ChatGoogleGenerativeAI,
|
12 |
+
HarmBlockThreshold,
|
13 |
+
HarmCategory,
|
14 |
+
)
|
15 |
+
import streamlit as st
|
16 |
+
|
17 |
+
from planing_tools import weatherapi_forecast_periods, query_rag
|
18 |
+
from calendar_tools import list_calendar_list, list_calendar_events, insert_calendar_event, create_calendar
|
19 |
+
from dotenv import load_dotenv
|
20 |
+
|
21 |
+
hoje = datetime.today()
|
22 |
+
|
23 |
+
data_formatada = hoje.strftime("%Y/%m/%d")
|
24 |
+
# Pega o dia da semana em inglês
|
25 |
+
dia_da_semana_ingles = hoje.strftime("%A")
|
26 |
+
|
27 |
+
# Dicionário para mapear os dias da semana de inglês para português
|
28 |
+
dias_da_semana_pt = {
|
29 |
+
"Monday": "segunda-feira",
|
30 |
+
"Tuesday": "terça-feira",
|
31 |
+
"Wednesday": "quarta-feira",
|
32 |
+
"Thursday": "quinta-feira",
|
33 |
+
"Friday": "sexta-feira",
|
34 |
+
"Saturday": "sábado",
|
35 |
+
"Sunday": "domingo"
|
36 |
+
}
|
37 |
+
|
38 |
+
# Traduz o dia da semana para português
|
39 |
+
dia_da_semana_pt = dias_da_semana_pt.get(dia_da_semana_ingles, dia_da_semana_ingles)
|
40 |
+
|
41 |
+
# Formata a data para utilizar no prompt
|
42 |
+
data_atual = f"{dia_da_semana_pt}, {data_formatada}"
|
43 |
+
|
44 |
+
load_dotenv()
|
45 |
+
os.environ["GOOGLE_API_KEY"] = os.getenv('GOOGLE_API_KEY')
|
46 |
+
os.environ["LANGCHAIN_API_KEY"] = os.getenv('LANGCHAIN_API_KEY')
|
47 |
+
|
48 |
+
llm = ChatGoogleGenerativeAI(
|
49 |
+
model="gemini-1.5-flash",
|
50 |
+
convert_system_message_to_human=True,
|
51 |
+
handle_parsing_errors=True,
|
52 |
+
temperature=0.6,
|
53 |
+
max_tokens= 1000,
|
54 |
+
safety_settings = {
|
55 |
+
HarmCategory.HARM_CATEGORY_HARASSMENT: HarmBlockThreshold.BLOCK_ONLY_HIGH,
|
56 |
+
HarmCategory.HARM_CATEGORY_HATE_SPEECH: HarmBlockThreshold.BLOCK_ONLY_HIGH,
|
57 |
+
HarmCategory.HARM_CATEGORY_SEXUALLY_EXPLICIT: HarmBlockThreshold.BLOCK_ONLY_HIGH,
|
58 |
+
HarmCategory.HARM_CATEGORY_DANGEROUS_CONTENT: HarmBlockThreshold.BLOCK_ONLY_HIGH,
|
59 |
+
},
|
60 |
+
)
|
61 |
+
|
62 |
+
def transfer_to_calendar_agent(input_str):
|
63 |
+
return calendar_agent_executor.invoke({"input": input_str})
|
64 |
+
def transfer_to_travel_agent(input_str):
|
65 |
+
return travel_agent_executor.invoke({"input": input_str})
|
66 |
+
|
67 |
+
ddg_search = DuckDuckGoSearchAPIWrapper()
|
68 |
+
|
69 |
+
travel_planing_tools = [
|
70 |
+
Tool(
|
71 |
+
name="DuckDuckGo Search",
|
72 |
+
func=ddg_search.run,
|
73 |
+
description="""Essa ferramenta DEVE ser utilizada para buscar eventos relevantes no período fornecido pelo usuário.
|
74 |
+
Ela é útil para obter informações sobre eventos ou atividades especiais que estão acontecendo na cidade de destino nas datas que o usuário informou.
|
75 |
+
O modelo deve usá-la para complementar as sugestões de atividades."""
|
76 |
+
),
|
77 |
+
Tool(
|
78 |
+
name="Weather Forecast",
|
79 |
+
func=lambda date_string: weatherapi_forecast_periods(date_string, st.session_state.selected_destino),
|
80 |
+
description="""Esta ferramenta DEVE ser usada obrigatoriamente *antes* de gerar o roteiro turístico, e somente após coletar todas as informações necessárias do usuário, incluindo o intervalo exato de datas.
|
81 |
+
A consulta do clima deve ser feita separadamente para cada dia do período informado, garantindo que as atividades planejadas no roteiro sejam compatíveis com as condições climáticas previstas.
|
82 |
+
|
83 |
+
**Quando usar:**
|
84 |
+
- O clima é um fator relevante para a definição de atividades no roteiro.
|
85 |
+
- O usuário especificou o intervalo de datas para o planejamento.
|
86 |
+
|
87 |
+
**Instruções:**
|
88 |
+
1. Certifique-se de coletar as datas exatas do período solicitado pelo usuário.
|
89 |
+
2. Consulte a previsão do tempo para cada dia separadamente no intervalo de datas fornecido.
|
90 |
+
3. Use o resultado da previsão para ajustar o planejamento das atividades de acordo com as condições climáticas.
|
91 |
+
|
92 |
+
**Exemplo de uso:**
|
93 |
+
- Entrada do usuário: "Planeje um roteiro entre os dias 1º de agosto e 4 de março."
|
94 |
+
- Ações do modelo:
|
95 |
+
Action input: 2025/08/01
|
96 |
+
Action input: 2025/08/02
|
97 |
+
Action input: 2025/08/03
|
98 |
+
Action input: 2025/08/04
|
99 |
+
"""
|
100 |
+
),
|
101 |
+
Tool(
|
102 |
+
name="Query RAG",
|
103 |
+
func=lambda query_text: query_rag(query_text, st.session_state.selected_destino),
|
104 |
+
description="""Esta ferramenta deve ser usada quando o modelo souber a cidade de destino e os interesses do usuário, com o objetivo de fornecer informações sobre pontos turísticos e atrações que se alinham com esses interesses.
|
105 |
+
O modelo deve utilizar essa ferramenta para sugerir atividades e lugares específicos a visitar, baseados na cidade e nos interesses fornecidos."""
|
106 |
+
),
|
107 |
+
Tool(
|
108 |
+
name="Calendar Agent",
|
109 |
+
func=transfer_to_calendar_agent,
|
110 |
+
description="Esse agente lida com tudo relacionado ao calendário. As mensagens enviadas a ele devem estar em Português."
|
111 |
+
),
|
112 |
+
]
|
113 |
+
from calendar_tools import list_calendar_list, list_calendar_events, insert_calendar_event, create_calendar
|
114 |
+
|
115 |
+
google_calendar_tools = [
|
116 |
+
Tool(
|
117 |
+
name="Create Calendar List",
|
118 |
+
func=create_calendar,
|
119 |
+
description="""
|
120 |
+
Use a função create_calendar_list para criar um novo calendário.
|
121 |
+
Forneça o resumo (título) do calendário que você deseja criar.
|
122 |
+
- Exemplo de uso: 'Meu Calendário de Viagens'
|
123 |
+
- Essa função vai criar um novo calendário com o resumo fornecido.
|
124 |
+
"""
|
125 |
+
),
|
126 |
+
Tool(
|
127 |
+
name="List Calendar List",
|
128 |
+
func=list_calendar_list,
|
129 |
+
description="""
|
130 |
+
Use a função list_calendar_list para obter a lista dos seus calendários disponíveis no Google Calendar.
|
131 |
+
Se você quiser limitar o número de calendários retornados, forneça o número máximo desejado.
|
132 |
+
- Exemplo de uso: Forneça o número máximo de calendários a serem listados, como 50. Se não quiser um limite, apenas diga para listar 200 o máximo por padrão.
|
133 |
+
"""
|
134 |
+
),
|
135 |
+
Tool(
|
136 |
+
name="List Calendar Events",
|
137 |
+
func=list_calendar_events,
|
138 |
+
description="""
|
139 |
+
Use a função list_calendar_events para obter a lista de eventos de um calendário específico.
|
140 |
+
Você precisa fornecer o ID do calendário e, opcionalmente, o número máximo de eventos a serem listados.
|
141 |
+
- Exemplo de uso para o calendário principal: forneça 'primary' como o ID do calendário.
|
142 |
+
- Exemplo de uso com limite: forneça o ID do calendário e o número máximo de eventos, como 'primary', 20.
|
143 |
+
- Para listar eventos de um calendário específico, primeiro liste seus calendários para encontrar o ID correto.
|
144 |
+
"""
|
145 |
+
),
|
146 |
+
Tool(
|
147 |
+
name="Insert Calendar Event",
|
148 |
+
func=insert_calendar_event,
|
149 |
+
description="""
|
150 |
+
Use a função Insert Calendar Event para adicionar **um único evento** a um calendário Google.
|
151 |
+
Esta função deve ser chamada **uma vez para cada evento** que precisa ser agendado.
|
152 |
+
|
153 |
+
A entrada para esta função deve ser uma descrição clara e completa de **um único evento**
|
154 |
+
em linguagem natural, contendo todas as informações necessárias para agendá-lo individualmente.
|
155 |
+
|
156 |
+
Se o usuário pediu para criar um novo calendário, você DEVE utilizar o calendar_id desse novo calendário para a inserção dos eventos.
|
157 |
+
|
158 |
+
**Formato de entrada esperado:**
|
159 |
+
|
160 |
+
**Descrição em Linguagem Natural (obrigatório):**
|
161 |
+
Forneça uma descrição textual detalhada para **um único evento**. Inclua o calendar_id (se fornecido pelo usuário), o que será feito,
|
162 |
+
quando (data e hora específicas), onde (local exato) e quaisquer detalhes adicionais relevantes
|
163 |
+
para esse evento em particular.
|
164 |
+
|
165 |
+
**Exemplos de descrições de eventos individuais:**
|
166 |
+
- "Agendar almoço com a Maria na sexta-feira ao meio-dia no restaurante X."
|
167 |
+
- "Criar um compromisso para pagar as contas no dia 10 de janeiro às 9h da manhã."
|
168 |
+
- "Bloquear minha agenda para foco no projeto Y na próxima segunda-feira, das 14h às 17h, na minha mesa."
|
169 |
+
|
170 |
+
**Como usar esta função corretamente:**
|
171 |
+
|
172 |
+
1. **Identifique cada evento individualmente:** Se a intenção é agendar múltiplos eventos,
|
173 |
+
trate cada um como uma chamada separada para esta função.
|
174 |
+
2. **Obtenha o ID do calendário (se necessário):** Se o usuário mencionar um nome de
|
175 |
+
calendário específico, use a função 'list_calendar_list' **primeiro** para obter
|
176 |
+
o 'calendar_id' correspondente.
|
177 |
+
3. **Chame esta função com a descrição de um único evento:** Forneça a descrição completa
|
178 |
+
em linguagem natural para o evento específico que você deseja agendar.
|
179 |
+
|
180 |
+
**Informações essenciais para cada descrição de evento:**
|
181 |
+
- **O que:** Uma descrição concisa do evento (ex: "Reunião de equipe", "Consulta médica").
|
182 |
+
- **Quando:** A data e hora de início e fim **precisas** do evento.
|
183 |
+
- **Onde:** O local **específico** do evento.
|
184 |
+
- **Fuso Horário (opcional):** Se aplicável, mencione o fuso horário se for diferente
|
185 |
+
de "America/Fortaleza".
|
186 |
+
|
187 |
+
**Exemplo de uso**
|
188 |
+
|
189 |
+
*Usuário: Agendar visita ao Aquário Natal sábado das 9h às 12h.*
|
190 |
+
*Agente:* (Chama 'Insert Calendar Event' com a Action Input: "Agendar visita ao Aquário Natal sábado das 9h às 12h.")
|
191 |
+
|
192 |
+
**Observação IMPORTANTE:** Esta função **NÃO** deve ser usada para criar múltiplos eventos
|
193 |
+
com uma única chamada. Cada chamada deve corresponder a **um único evento**.
|
194 |
+
|
195 |
+
Ao final da execução desta função, você DEVE fornecer ao usuário a confirmação do agendamento
|
196 |
+
desse evento específico, incluindo o título, data, hora.
|
197 |
+
|
198 |
+
No final forneça o link para o calendário google: [https://www.google.com/calendar]
|
199 |
+
"""
|
200 |
+
)
|
201 |
+
]
|
202 |
+
|
203 |
+
travel_planing_prompt = hub.pull("tales/agente_turismo")
|
204 |
+
google_calendar_prompt = hub.pull("tales/agente_calendario")
|
205 |
+
|
206 |
+
llm_with_stop = llm.bind(stop=["\nObservation"])
|
207 |
+
|
208 |
+
planing_prompt = travel_planing_prompt.partial(
|
209 |
+
tools=render_text_description(travel_planing_tools),
|
210 |
+
tool_names=", ".join([t.name for t in travel_planing_tools]),
|
211 |
+
)
|
212 |
+
|
213 |
+
calendar_prompt = google_calendar_prompt.partial(
|
214 |
+
tools=render_text_description(google_calendar_tools),
|
215 |
+
tool_names=", ".join([t.name for t in google_calendar_tools]),
|
216 |
+
)
|
217 |
+
|
218 |
+
history = ChatMessageHistory()
|
219 |
+
memory = ConversationBufferWindowMemory(
|
220 |
+
k=20,
|
221 |
+
chat_memory=history,
|
222 |
+
memory_key="chat_history",
|
223 |
+
input_key="input",
|
224 |
+
other_memory_key=["destino"])
|
225 |
+
|
226 |
+
travel_planing_agent = (
|
227 |
+
{
|
228 |
+
"input": lambda x: x["input"],
|
229 |
+
"destino": lambda x: x.get("destino"),
|
230 |
+
"agent_scratchpad": lambda x: format_log_to_str(x["intermediate_steps"]),
|
231 |
+
"chat_history": lambda x: x["chat_history"],
|
232 |
+
"data_atual": lambda x: data_atual,
|
233 |
+
}
|
234 |
+
| planing_prompt
|
235 |
+
| llm_with_stop
|
236 |
+
| ReActSingleInputOutputParser()
|
237 |
+
)
|
238 |
+
|
239 |
+
travel_agent_executor = AgentExecutor(agent=travel_planing_agent, tools=travel_planing_tools, verbose=True, memory=memory, handle_parsing_errors=True)
|
240 |
+
|
241 |
+
google_calendar_agent = (
|
242 |
+
{
|
243 |
+
"input": lambda x: x["input"],
|
244 |
+
"agent_scratchpad": lambda x: format_log_to_str(x["intermediate_steps"]),
|
245 |
+
"chat_history": lambda x: x["chat_history"],
|
246 |
+
"data_atual": lambda x: data_atual,
|
247 |
+
}
|
248 |
+
| calendar_prompt
|
249 |
+
| llm_with_stop
|
250 |
+
| ReActSingleInputOutputParser()
|
251 |
+
)
|
252 |
+
|
253 |
+
calendar_agent_executor = AgentExecutor(agent=google_calendar_agent, tools=google_calendar_tools, verbose=True, memory=memory)
|
app.py
ADDED
@@ -0,0 +1,44 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import streamlit as st
|
2 |
+
from agents import travel_agent_executor
|
3 |
+
from unidecode import unidecode
|
4 |
+
|
5 |
+
DESTINOS = {
|
6 |
+
"Natal": "natal",
|
7 |
+
"Caicó": "caico",
|
8 |
+
"Pipa": "pipa"
|
9 |
+
}
|
10 |
+
|
11 |
+
if __name__ == '__main__':
|
12 |
+
|
13 |
+
st.title('Agente Turístico')
|
14 |
+
st.sidebar.title('Escolha um destino')
|
15 |
+
|
16 |
+
destino_selecionado = st.sidebar.selectbox('Destino', list(DESTINOS.keys()))
|
17 |
+
|
18 |
+
if destino_selecionado:
|
19 |
+
st.write(f"Você selecionou {destino_selecionado}")
|
20 |
+
st.session_state.selected_destino = DESTINOS[destino_selecionado]
|
21 |
+
|
22 |
+
if 'messages' not in st.session_state:
|
23 |
+
st.session_state.messages = []
|
24 |
+
|
25 |
+
for message in st.session_state.messages:
|
26 |
+
with st.chat_message(message['role']):
|
27 |
+
st.markdown(message['content'])
|
28 |
+
|
29 |
+
if prompt := st.chat_input('Digite a sua mensagem.'):
|
30 |
+
st.session_state.messages.append({'role': 'user', 'content': prompt})
|
31 |
+
|
32 |
+
with st.chat_message('user', avatar='🧑💻'):
|
33 |
+
st.markdown(prompt)
|
34 |
+
|
35 |
+
with st.chat_message('ai', avatar='🤖'):
|
36 |
+
# Passa o destino selecionado como contexto para o agente
|
37 |
+
response = travel_agent_executor.invoke({
|
38 |
+
"input": prompt,
|
39 |
+
"destino": unidecode(DESTINOS[destino_selecionado].lower())
|
40 |
+
})
|
41 |
+
agent_response = response['output']
|
42 |
+
st.markdown(agent_response)
|
43 |
+
|
44 |
+
st.session_state.messages.append({'role': 'assistant', 'content': agent_response})
|
calendar_tools.py
ADDED
@@ -0,0 +1,236 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import json
|
2 |
+
import time
|
3 |
+
from pydantic import BaseModel, Field, ValidationError
|
4 |
+
from typing import List
|
5 |
+
from langchain.output_parsers import PydanticOutputParser
|
6 |
+
from langchain.prompts import ChatPromptTemplate
|
7 |
+
from google_apis import create_service
|
8 |
+
from langchain_google_genai import (
|
9 |
+
ChatGoogleGenerativeAI,
|
10 |
+
HarmBlockThreshold,
|
11 |
+
HarmCategory,
|
12 |
+
)
|
13 |
+
client_secret = 'client_secret.json'
|
14 |
+
|
15 |
+
class Evento(BaseModel):
|
16 |
+
calendar_id: str = Field(default="primary", description="Identificador do calendário do Google Agenda onde o evento será criado. Por padrão, utiliza o calendário principal do usuário ('primary'). Para usar um calendário diferente, forneça o ID do calendário específico.")
|
17 |
+
summary: str = Field(description="Título ou resumo conciso do evento, que será exibido no Google Agenda.")
|
18 |
+
location: str = Field(description="Localização do evento. Pode ser um endereço físico ou um nome de local.")
|
19 |
+
description: str = Field(description="Descrição detalhada do evento, incluindo informações adicionais, agenda, objetivos.")
|
20 |
+
start: str = Field(description="Data e hora de início do evento no formato ISO 8601 (Exemplo: '2023-10-27T10:00:00').")
|
21 |
+
end: str = Field(description="Data e hora de término do evento no formato ISO 8601 (Exemplo: '2023-10-27T11:00:00').")
|
22 |
+
timezone: str = Field(default="America/Fortaleza", description="Fuso horário do evento, utilizando a nomenclatura da IANA Time Zone Database (e.g., 'America/Sao_Paulo', 'Europe/London'). O padrão é 'America/Fortaleza'.")
|
23 |
+
|
24 |
+
def construct_google_calendar_client(client_secret):
|
25 |
+
"""
|
26 |
+
Constrói um cliente para a API Google Calendar.
|
27 |
+
|
28 |
+
Parâmetros:
|
29 |
+
- client_secret (str): O caminho para o arquivo client secret JSON.
|
30 |
+
|
31 |
+
Retorna:
|
32 |
+
- service: A instância de serviço com o API do Google Calendar.
|
33 |
+
"""
|
34 |
+
|
35 |
+
API_NAME = 'calendar'
|
36 |
+
API_VERSION = 'v3'
|
37 |
+
SCOPES = ['https://www.googleapis.com/auth/calendar']
|
38 |
+
service = create_service(client_secret, API_NAME, API_VERSION, SCOPES)
|
39 |
+
return service
|
40 |
+
|
41 |
+
calendar_service = construct_google_calendar_client(client_secret)
|
42 |
+
|
43 |
+
def create_calendar(calendar_name):
|
44 |
+
"""
|
45 |
+
Criar uma nova lista de calendários.
|
46 |
+
|
47 |
+
Parâmetros:
|
48 |
+
- calendar_name (str): O nome da nova lista de calendário.
|
49 |
+
|
50 |
+
Retorno:
|
51 |
+
- dict: Um dicionário contendo o ID da nova lista de calendário.
|
52 |
+
"""
|
53 |
+
calendar_name = calendar_name.replace("'", "").replace('"', "")
|
54 |
+
calendar_list = {
|
55 |
+
'summary': calendar_name
|
56 |
+
}
|
57 |
+
created_calendar_list = calendar_service.calendars().insert(body=calendar_list).execute()
|
58 |
+
return created_calendar_list
|
59 |
+
|
60 |
+
def list_calendar_list(max_capacity=200):
|
61 |
+
"""
|
62 |
+
Lista todas as listas de calendários disponíveis, com limite de capacidade.
|
63 |
+
|
64 |
+
Parâmetros:
|
65 |
+
- max_capacity (int, opcional): O número máximo de calendários a serem retornados.
|
66 |
+
Padrão: 200.
|
67 |
+
|
68 |
+
Retorna:
|
69 |
+
- list: Uma lista de dicionários, onde cada dicionário contém os detalhes
|
70 |
+
de um calendário (ID, nome e descrição).
|
71 |
+
"""
|
72 |
+
if isinstance(max_capacity, str):
|
73 |
+
max_capacity = int(max_capacity)
|
74 |
+
|
75 |
+
all_calendars = []
|
76 |
+
all_calendars_cleaned = []
|
77 |
+
next_page_token = None
|
78 |
+
capacity_tracker = 0
|
79 |
+
|
80 |
+
while True:
|
81 |
+
calendar_list = calendar_service.calendarList().list(
|
82 |
+
maxResults = min(200, max_capacity - capacity_tracker),
|
83 |
+
pageToken = next_page_token
|
84 |
+
).execute()
|
85 |
+
calendars = calendar_list.get('items', [])
|
86 |
+
all_calendars.extend(calendars)
|
87 |
+
capacity_tracker += len(calendars)
|
88 |
+
if capacity_tracker >= max_capacity:
|
89 |
+
break
|
90 |
+
next_page_token = calendar_list.get('nextPageToken')
|
91 |
+
if not next_page_token:
|
92 |
+
break
|
93 |
+
|
94 |
+
for calendar in all_calendars:
|
95 |
+
all_calendars_cleaned.append(
|
96 |
+
{
|
97 |
+
'id': calendar['id'],
|
98 |
+
'name': calendar['summary'],
|
99 |
+
'description': calendar.get('description', '')
|
100 |
+
})
|
101 |
+
return all_calendars_cleaned
|
102 |
+
|
103 |
+
def list_calendar_events(calendar_id, max_capacity=20):
|
104 |
+
"""
|
105 |
+
Lista os eventos de um calendário específico.
|
106 |
+
|
107 |
+
Parâmetros:
|
108 |
+
- calendar_id (str): O ID do calendário para listar os eventos.
|
109 |
+
- max_capacity (int, opcional): O número máximo de eventos a serem retornados.
|
110 |
+
Padrão: 20.
|
111 |
+
|
112 |
+
Retorna:
|
113 |
+
- list: Uma lista de dicionários contendo os detalhes dos eventos.
|
114 |
+
"""
|
115 |
+
if isinstance(max_capacity, str):
|
116 |
+
max_capacity = int(max_capacity)
|
117 |
+
|
118 |
+
all_events = []
|
119 |
+
next_page_token = None
|
120 |
+
capacity_tracker = 0
|
121 |
+
while True:
|
122 |
+
events_list = calendar_service.events().list(
|
123 |
+
calendarId=calendar_id,
|
124 |
+
maxResults=min(250, max_capacity - capacity_tracker),
|
125 |
+
pageToken=next_page_token
|
126 |
+
).execute()
|
127 |
+
events = events_list.get('items', [])
|
128 |
+
all_events.extend(events)
|
129 |
+
capacity_tracker += len(events)
|
130 |
+
if capacity_tracker >= max_capacity:
|
131 |
+
break
|
132 |
+
if not next_page_token:
|
133 |
+
break
|
134 |
+
return all_events
|
135 |
+
|
136 |
+
def extract_event_parameters(query):
|
137 |
+
"""
|
138 |
+
Extrai parâmetros de um evento a partir de uma descrição em linguagem natural.
|
139 |
+
|
140 |
+
Esta função utiliza um modelo de linguagem grande para interpretar a descrição do evento
|
141 |
+
fornecida e extrair informações como resumo, local, descrição, horário de início e fim,
|
142 |
+
e fuso horário. A resposta é formatada como um JSON de acordo com o esquema definido pela
|
143 |
+
classe `Evento`.
|
144 |
+
|
145 |
+
Args:
|
146 |
+
query (str): A descrição do evento em linguagem natural.
|
147 |
+
Exemplo: "Marcar reunião com o time de marketing amanhã às 10h na sala de conferência."
|
148 |
+
|
149 |
+
Returns:
|
150 |
+
str: Uma string JSON contendo os parâmetros extraídos do evento, ou None se a extração falhar.
|
151 |
+
O JSON terá a seguinte estrutura:
|
152 |
+
{
|
153 |
+
"calendar_id": "ID do calendário Google (padrão: 'primary')",
|
154 |
+
"summary": "Resumo do evento",
|
155 |
+
"location": "Local do evento",
|
156 |
+
"description": "Descrição do evento",
|
157 |
+
"start": "Hora de início do evento",
|
158 |
+
"end": "Hora de término do evento",
|
159 |
+
"timezone": "Fuso horário do evento (padrão: 'America/Fortaleza')"
|
160 |
+
}
|
161 |
+
"""
|
162 |
+
llm = ChatGoogleGenerativeAI(
|
163 |
+
model="gemini-1.5-flash",
|
164 |
+
convert_system_message_to_human=True,
|
165 |
+
handle_parsing_errors=True,
|
166 |
+
temperature=0.6,
|
167 |
+
max_tokens= 1000,
|
168 |
+
safety_settings = {
|
169 |
+
HarmCategory.HARM_CATEGORY_HARASSMENT: HarmBlockThreshold.BLOCK_ONLY_HIGH,
|
170 |
+
HarmCategory.HARM_CATEGORY_HATE_SPEECH: HarmBlockThreshold.BLOCK_ONLY_HIGH,
|
171 |
+
HarmCategory.HARM_CATEGORY_SEXUALLY_EXPLICIT: HarmBlockThreshold.BLOCK_ONLY_HIGH,
|
172 |
+
HarmCategory.HARM_CATEGORY_DANGEROUS_CONTENT: HarmBlockThreshold.BLOCK_ONLY_HIGH,
|
173 |
+
},
|
174 |
+
)
|
175 |
+
parser = PydanticOutputParser(pydantic_object=Evento)
|
176 |
+
|
177 |
+
prompt = ChatPromptTemplate.from_messages([
|
178 |
+
("system", "Você é um assistente especialista em extrair informações de eventos de textos. Extraia os parâmetros necessários para criar um evento, incluindo data e hora de início e fim. Retorne a resposta formatada em JSON de acordo com o esquema fornecido.\n\n{format_instructions}\n"),
|
179 |
+
("human", "{user_query}")
|
180 |
+
])
|
181 |
+
chain = prompt | llm | parser
|
182 |
+
|
183 |
+
resposta = chain.invoke({
|
184 |
+
"language": "Portuguese",
|
185 |
+
"user_query": query,
|
186 |
+
"format_instructions": parser.get_format_instructions(),
|
187 |
+
})
|
188 |
+
resposta_dict = {
|
189 |
+
'summary': resposta.summary,
|
190 |
+
'location': resposta.location,
|
191 |
+
'description': resposta.description,
|
192 |
+
'start': {
|
193 |
+
"dateTime": resposta.start,
|
194 |
+
"timeZone": resposta.timezone
|
195 |
+
},
|
196 |
+
'end': {
|
197 |
+
"dateTime": resposta.end,
|
198 |
+
"timeZone": resposta.timezone
|
199 |
+
},
|
200 |
+
'attendees': []
|
201 |
+
}
|
202 |
+
|
203 |
+
|
204 |
+
resposta_json = json.dumps(resposta_dict, ensure_ascii=False)
|
205 |
+
return resposta.calendar_id, resposta_json
|
206 |
+
def insert_calendar_event(event):
|
207 |
+
"""
|
208 |
+
Insere um evento em um calendário Google.
|
209 |
+
|
210 |
+
Parâmetros:
|
211 |
+
- event (str): Uma descrição em linguagem natural do evento a ser criado.
|
212 |
+
A função `extract_event_parameters` será utilizada para extrair os detalhes
|
213 |
+
do evento a partir desta descrição e convertê-los em um objeto `Evento`.
|
214 |
+
Exemplos:
|
215 |
+
- "Agendar almoço com a Maria na sexta-feira ao meio-dia no restaurante X."
|
216 |
+
- "Criar um lembrete para pagar as contas no dia 10 às 9h da manhã."
|
217 |
+
- "Bloquear minha agenda para foco no projeto Y na próxima segunda das 14h às 17h."
|
218 |
+
|
219 |
+
Retorna:
|
220 |
+
- dict: Detalhes do evento criado no Google Agenda ou mensagem de erro.
|
221 |
+
Em caso de sucesso, o dicionário conterá informações sobre o evento criado,
|
222 |
+
incluindo o ID do evento. Em caso de erro, retornará uma mensagem descrevendo
|
223 |
+
o problema encontrado durante o processamento ou inserção do evento.
|
224 |
+
"""
|
225 |
+
extracted_data = extract_event_parameters(event)
|
226 |
+
if not extracted_data:
|
227 |
+
return "Erro: Não foi possível extrair os detalhes do evento da descrição fornecida."
|
228 |
+
|
229 |
+
calendar_id, event_details = extracted_data
|
230 |
+
request_body = json.loads(event_details)
|
231 |
+
time.sleep(1) ## Evitar limites de requisições do plano gratuito
|
232 |
+
event = calendar_service.events().insert(
|
233 |
+
calendarId = calendar_id,
|
234 |
+
body = request_body
|
235 |
+
).execute()
|
236 |
+
return event
|
chroma/caico/chroma.sqlite3
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
version https://git-lfs.github.com/spec/v1
|
2 |
+
oid sha256:793ffe1eb8a916a47add32957635f0a2007ea3ebc564f42e0bae192a0d594a69
|
3 |
+
size 3612672
|
chroma/natal/chroma.sqlite3
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
version https://git-lfs.github.com/spec/v1
|
2 |
+
oid sha256:fcdeb0d0b040befdf4efdf50edbafacf81d6e8bfdbf8b33ed6e98f416692d34c
|
3 |
+
size 3563520
|
chroma/pipa/chroma.sqlite3
ADDED
Binary file (492 kB). View file
|
|
get_embedding_function.py
ADDED
@@ -0,0 +1,6 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from langchain_huggingface import HuggingFaceEmbeddings
|
2 |
+
|
3 |
+
def get_embedding_function():
|
4 |
+
model_name = 'Snowflake/snowflake-arctic-embed-l-v2.0'
|
5 |
+
embeddings = HuggingFaceEmbeddings(model_name=model_name)
|
6 |
+
return embeddings
|
google_apis.py
ADDED
@@ -0,0 +1,43 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
from google_auth_oauthlib.flow import InstalledAppFlow
|
3 |
+
from googleapiclient.discovery import build
|
4 |
+
from google.oauth2.credentials import Credentials
|
5 |
+
from google.auth.transport.requests import Request
|
6 |
+
|
7 |
+
def create_service(client_secret_file, api_name, api_version, *scopes, prefix=''):
|
8 |
+
CLIENT_SECRET_FILE = client_secret_file
|
9 |
+
API_SERVICE_NAME = api_name
|
10 |
+
API_VERSION = api_version
|
11 |
+
SCOPES = [scope for scope in scopes[0]]
|
12 |
+
|
13 |
+
creds = None
|
14 |
+
working_dir = os.getcwd()
|
15 |
+
token_dir = 'token files'
|
16 |
+
token_file = f'token_{API_SERVICE_NAME}_{API_VERSION}{prefix}.json'
|
17 |
+
|
18 |
+
### Check if token dir exists first, if notm create the folder
|
19 |
+
if not os.path.exists(os.path.join(working_dir, token_dir)):
|
20 |
+
os.mkdir(os.path.join(working_dir, token_dir))
|
21 |
+
|
22 |
+
if os.path.exists(os.path.join(working_dir,token_dir, token_file)):
|
23 |
+
creds = Credentials.from_authorized_user_file(os.path.join(working_dir, token_dir, token_file), SCOPES)
|
24 |
+
|
25 |
+
if not creds or not creds.valid:
|
26 |
+
if creds and creds.expired and creds.refresh_token:
|
27 |
+
creds.refresh(Request())
|
28 |
+
else:
|
29 |
+
flow = InstalledAppFlow.from_client_secrets_file(CLIENT_SECRET_FILE, SCOPES)
|
30 |
+
creds = flow.run_local_server(port=0)
|
31 |
+
|
32 |
+
with open(os.path.join(working_dir, token_dir, token_file), 'w') as token:
|
33 |
+
token.write(creds.to_json())
|
34 |
+
|
35 |
+
try:
|
36 |
+
service = build(API_SERVICE_NAME, API_VERSION, credentials=creds, static_discovery=False)
|
37 |
+
print(API_SERVICE_NAME, API_VERSION, 'service created successfully')
|
38 |
+
return service
|
39 |
+
except Exception as e:
|
40 |
+
print(e)
|
41 |
+
print(f'Failed to create service instance for {API_SERVICE_NAME}')
|
42 |
+
os.remove(os.path.join(working_dir, token_dir, token_file))
|
43 |
+
return None
|
planing_tools.py
ADDED
@@ -0,0 +1,89 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from dotenv import load_dotenv
|
2 |
+
import os
|
3 |
+
import requests
|
4 |
+
from get_embedding_function import get_embedding_function
|
5 |
+
from langchain_chroma import Chroma
|
6 |
+
|
7 |
+
load_dotenv()
|
8 |
+
|
9 |
+
|
10 |
+
CHROMA_PATH = "chroma"
|
11 |
+
WEATHER_API = os.getenv('WEATHER_API')
|
12 |
+
BASE_URL = "http://api.weatherapi.com/v1/forecast.json"
|
13 |
+
|
14 |
+
def weatherapi_forecast_periods(date_string: str, destino: str) -> str:
|
15 |
+
"""
|
16 |
+
Obtém a previsão do tempo para a cidade da cidade destino em uma data específica,
|
17 |
+
separada em manhã, tarde e noite.
|
18 |
+
|
19 |
+
Args:
|
20 |
+
date_string (str): Uma string contendo a data no formato yyyy-mm-dd.
|
21 |
+
|
22 |
+
Returns:
|
23 |
+
str: Uma string contendo as previsões separadas por períodos para a data especificada.
|
24 |
+
"""
|
25 |
+
try:
|
26 |
+
params = {
|
27 |
+
"key": WEATHER_API,
|
28 |
+
"q": destino.capitalize(),
|
29 |
+
"dt": date_string,
|
30 |
+
"aqi": "no",
|
31 |
+
"alerts": "no",
|
32 |
+
"lang": "pt"
|
33 |
+
}
|
34 |
+
response = requests.get(BASE_URL, params=params)
|
35 |
+
response.raise_for_status()
|
36 |
+
data = response.json()
|
37 |
+
|
38 |
+
if data and data.get("forecast") and data["forecast"].get("forecastday"):
|
39 |
+
forecast_data = data["forecast"]["forecastday"][0]
|
40 |
+
hourly_data = forecast_data.get("hour", [])
|
41 |
+
|
42 |
+
periods = {
|
43 |
+
"Manhã": range(5, 12),
|
44 |
+
"Tarde": range(13, 18),
|
45 |
+
"Noite": range(18, 24)
|
46 |
+
}
|
47 |
+
|
48 |
+
result = f"Previsão para {date_string} em {destino.capitalize()}:\n"
|
49 |
+
for period, hours in periods.items():
|
50 |
+
result += f"\n{period}:\n"
|
51 |
+
filtered_data = [
|
52 |
+
{
|
53 |
+
"hora": hour["time"].split(" ")[1],
|
54 |
+
"temperatura": hour["temp_c"],
|
55 |
+
"condicao": hour["condition"]["text"],
|
56 |
+
"chance_de_chuva": hour["chance_of_rain"],
|
57 |
+
"umidade": hour["humidity"]
|
58 |
+
}
|
59 |
+
for hour in hourly_data if int(hour["time"].split(" ")[1].split(":")[0]) in hours
|
60 |
+
]
|
61 |
+
if filtered_data:
|
62 |
+
for hour in filtered_data:
|
63 |
+
result += (
|
64 |
+
f"Hora: {hour['hora']} - "
|
65 |
+
f"Temp: {hour['temperatura']}°C, "
|
66 |
+
f"Condição: {hour['condicao']}, "
|
67 |
+
f"Chance de chuva: {hour['chance_de_chuva']}%, "
|
68 |
+
f"Umidade: {hour['umidade']}%\n"
|
69 |
+
)
|
70 |
+
else:
|
71 |
+
result += " Nenhuma previsão encontrada para este período.\n"
|
72 |
+
|
73 |
+
return result
|
74 |
+
else:
|
75 |
+
return "Previsão não encontrada para essa data."
|
76 |
+
except requests.exceptions.RequestException as e:
|
77 |
+
return f"Erro ao buscar informações meteorológicas: {str(e)}"
|
78 |
+
except Exception as e:
|
79 |
+
return f"Erro inesperado: {str(e)}"
|
80 |
+
|
81 |
+
def query_rag(query_text: str, destino: str) -> str:
|
82 |
+
embedding_function = get_embedding_function()
|
83 |
+
db = Chroma(persist_directory=f"{CHROMA_PATH}/{destino}", embedding_function=embedding_function)
|
84 |
+
|
85 |
+
|
86 |
+
results = db.similarity_search_with_score(f"{destino}: {query_text}", k=5)
|
87 |
+
|
88 |
+
context_text = "\n\n---\n\n".join([doc.page_content for doc, _score in results])
|
89 |
+
return context_text
|
populate_database.py
ADDED
@@ -0,0 +1,134 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import argparse
|
2 |
+
import os
|
3 |
+
import shutil
|
4 |
+
from langchain_community.document_loaders.pdf import PyPDFDirectoryLoader
|
5 |
+
from langchain_text_splitters import RecursiveCharacterTextSplitter
|
6 |
+
from langchain.schema.document import Document
|
7 |
+
from get_embedding_function import get_embedding_function
|
8 |
+
from langchain_chroma import Chroma
|
9 |
+
|
10 |
+
|
11 |
+
CHROMA_ROOT_PATH = "chroma"
|
12 |
+
DATA_ROOT_PATH = "pdf"
|
13 |
+
|
14 |
+
|
15 |
+
def main():
|
16 |
+
|
17 |
+
parser = argparse.ArgumentParser()
|
18 |
+
parser.add_argument("--reset", action="store_true", help="Reset the databases.")
|
19 |
+
args = parser.parse_args()
|
20 |
+
if args.reset:
|
21 |
+
print("Clearing all Chromas")
|
22 |
+
clear_all_databases()
|
23 |
+
|
24 |
+
|
25 |
+
for city_folder in os.listdir(DATA_ROOT_PATH):
|
26 |
+
city_path = os.path.join(DATA_ROOT_PATH, city_folder)
|
27 |
+
|
28 |
+
|
29 |
+
if os.path.isdir(city_path):
|
30 |
+
print(f"🔄 Processando a cidade: {city_folder}")
|
31 |
+
process_city(city_folder, city_path)
|
32 |
+
|
33 |
+
|
34 |
+
def process_city(city_name: str, city_path: str):
|
35 |
+
"""
|
36 |
+
Processa uma subpasta de cidade, criando ou atualizando o Chroma correspondente.
|
37 |
+
"""
|
38 |
+
|
39 |
+
chroma_city_path = os.path.join(CHROMA_ROOT_PATH, f"{city_name}")
|
40 |
+
|
41 |
+
|
42 |
+
documents = load_documents(city_path)
|
43 |
+
chunks = split_documents(documents)
|
44 |
+
add_to_chroma(chunks, chroma_city_path)
|
45 |
+
|
46 |
+
|
47 |
+
def load_documents(city_path: str):
|
48 |
+
"""
|
49 |
+
Carrega documentos PDF da subpasta de uma cidade.
|
50 |
+
"""
|
51 |
+
document_loader = PyPDFDirectoryLoader(city_path)
|
52 |
+
return document_loader.load()
|
53 |
+
|
54 |
+
|
55 |
+
def split_documents(documents: list[Document]):
|
56 |
+
"""
|
57 |
+
Divide os documentos em chunks menores.
|
58 |
+
"""
|
59 |
+
text_splitter = RecursiveCharacterTextSplitter(
|
60 |
+
chunk_size=800,
|
61 |
+
chunk_overlap=80,
|
62 |
+
length_function=len,
|
63 |
+
is_separator_regex=False,
|
64 |
+
)
|
65 |
+
return text_splitter.split_documents(documents)
|
66 |
+
|
67 |
+
|
68 |
+
def add_to_chroma(chunks: list[Document], chroma_path: str):
|
69 |
+
"""
|
70 |
+
Adiciona ou atualiza os documentos no Chroma específico da cidade.
|
71 |
+
"""
|
72 |
+
|
73 |
+
db = Chroma(
|
74 |
+
persist_directory=chroma_path, embedding_function=get_embedding_function()
|
75 |
+
)
|
76 |
+
|
77 |
+
|
78 |
+
chunks_with_ids = calculate_chunk_ids(chunks)
|
79 |
+
|
80 |
+
|
81 |
+
existing_items = db.get(include=[])
|
82 |
+
existing_ids = set(existing_items["ids"])
|
83 |
+
print(f"Número de documentos no banco de dados '{chroma_path}': {len(existing_ids)}")
|
84 |
+
|
85 |
+
|
86 |
+
new_chunks = [chunk for chunk in chunks_with_ids if chunk.metadata["id"] not in existing_ids]
|
87 |
+
|
88 |
+
if len(new_chunks):
|
89 |
+
print(f"👉 Adicionando {len(new_chunks)} novo(s) documento(s) ao banco '{chroma_path}'")
|
90 |
+
new_chunk_ids = [chunk.metadata["id"] for chunk in new_chunks]
|
91 |
+
db.add_documents(new_chunks, ids=new_chunk_ids)
|
92 |
+
|
93 |
+
else:
|
94 |
+
print(f"✅ Nenhum novo documento para adicionar ao banco '{chroma_path}'")
|
95 |
+
|
96 |
+
|
97 |
+
def calculate_chunk_ids(chunks):
|
98 |
+
"""
|
99 |
+
Calcula IDs únicos para cada chunk com base na fonte e na página.
|
100 |
+
"""
|
101 |
+
last_page_id = None
|
102 |
+
current_chunk_index = 0
|
103 |
+
|
104 |
+
for chunk in chunks:
|
105 |
+
source = chunk.metadata.get("source")
|
106 |
+
page = chunk.metadata.get("page")
|
107 |
+
current_page_id = f"{source}:{page}"
|
108 |
+
|
109 |
+
|
110 |
+
if current_page_id == last_page_id:
|
111 |
+
current_chunk_index += 1
|
112 |
+
else:
|
113 |
+
current_chunk_index = 0
|
114 |
+
|
115 |
+
|
116 |
+
chunk_id = f"{current_page_id}:{current_chunk_index}"
|
117 |
+
last_page_id = current_page_id
|
118 |
+
|
119 |
+
|
120 |
+
chunk.metadata["id"] = chunk_id
|
121 |
+
|
122 |
+
return chunks
|
123 |
+
|
124 |
+
|
125 |
+
def clear_all_databases():
|
126 |
+
"""
|
127 |
+
Remove todos os bancos de dados Chroma existentes.
|
128 |
+
"""
|
129 |
+
if os.path.exists(CHROMA_ROOT_PATH):
|
130 |
+
shutil.rmtree(CHROMA_ROOT_PATH)
|
131 |
+
|
132 |
+
|
133 |
+
if __name__ == "__main__":
|
134 |
+
main()
|
requirements.txt
ADDED
@@ -0,0 +1,18 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
langchain==0.3.14
|
2 |
+
langchain-google-genai==2.0.9
|
3 |
+
langchain_chroma
|
4 |
+
langchain_community
|
5 |
+
langchain_huggingface
|
6 |
+
duckduckgo-search
|
7 |
+
chromadb==0.5.23
|
8 |
+
pydantic==2.10.5
|
9 |
+
pypdf==5.1.0
|
10 |
+
google-api-python-client==2.159.0
|
11 |
+
google-auth-httplib2==0.2.0
|
12 |
+
google-auth-oauthlib==1.2.1
|
13 |
+
streamlit==1.41.1
|
14 |
+
python-dotenv==1.0.1
|
15 |
+
moviepy==1.0.3
|
16 |
+
openai-whisper==20240930
|
17 |
+
reportlab==4.2.5
|
18 |
+
Unidecode==1.3.8
|
video_transcriptor.py
ADDED
@@ -0,0 +1,95 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from moviepy.editor import *
|
2 |
+
import whisper
|
3 |
+
import os
|
4 |
+
from reportlab.pdfgen import canvas
|
5 |
+
from reportlab.lib.pagesizes import letter
|
6 |
+
|
7 |
+
# Caminho da pasta principal com as subpastas de vídeos e onde os PDFs serão salvos
|
8 |
+
videos_root_folder = "pasta_com_videos" # Pasta principal contendo as subpastas por cidade
|
9 |
+
pdf_root_folder = "pdf" # Pasta principal para salvar os PDFs
|
10 |
+
|
11 |
+
# Criar a pasta PDF principal, caso não exista
|
12 |
+
if not os.path.exists(pdf_root_folder):
|
13 |
+
os.makedirs(pdf_root_folder)
|
14 |
+
|
15 |
+
# Carregar o modelo Whisper
|
16 |
+
modelo = whisper.load_model("base")
|
17 |
+
|
18 |
+
# Configurações da página PDF
|
19 |
+
page_width, page_height = letter
|
20 |
+
margin_left = 50
|
21 |
+
margin_top = 750
|
22 |
+
line_spacing = 14
|
23 |
+
line_length = 90 # Limite de caracteres por linha (ajuste conforme necessário)
|
24 |
+
|
25 |
+
# Iterar pelas subpastas dentro da pasta principal de vídeos
|
26 |
+
for city_folder in os.listdir(videos_root_folder):
|
27 |
+
city_path = os.path.join(videos_root_folder, city_folder)
|
28 |
+
|
29 |
+
if os.path.isdir(city_path): # Verificar se é uma pasta (cidade)
|
30 |
+
# Criar a subpasta correspondente na pasta PDF
|
31 |
+
city_pdf_folder = os.path.join(pdf_root_folder, city_folder)
|
32 |
+
if not os.path.exists(city_pdf_folder):
|
33 |
+
os.makedirs(city_pdf_folder)
|
34 |
+
|
35 |
+
# Contador para nomear os arquivos de transcrição dentro da cidade
|
36 |
+
contador = 1
|
37 |
+
|
38 |
+
# Iterar pelos arquivos de vídeo dentro da subpasta da cidade
|
39 |
+
for video_filename in os.listdir(city_path):
|
40 |
+
if video_filename.endswith(('.mp4', '.avi', '.mov')): # Filtrar apenas arquivos de vídeo
|
41 |
+
# Caminho completo do arquivo de vídeo
|
42 |
+
video_path = os.path.join(city_path, video_filename)
|
43 |
+
|
44 |
+
# Carregar o vídeo
|
45 |
+
video = VideoFileClip(video_path)
|
46 |
+
|
47 |
+
# Extrair áudio do vídeo e salvar como um arquivo WAV temporário
|
48 |
+
audio_path = "audio_extraido.wav"
|
49 |
+
video.audio.write_audiofile(audio_path, codec='pcm_s16le')
|
50 |
+
|
51 |
+
# Transcrever o áudio
|
52 |
+
result = modelo.transcribe(audio_path)
|
53 |
+
|
54 |
+
# Nome do arquivo PDF
|
55 |
+
pdf_filename = f"transcription_{contador}.pdf"
|
56 |
+
pdf_path = os.path.join(city_pdf_folder, pdf_filename)
|
57 |
+
|
58 |
+
# Criar o PDF e adicionar o texto transcrito
|
59 |
+
c = canvas.Canvas(pdf_path, pagesize=letter)
|
60 |
+
text = result["text"]
|
61 |
+
|
62 |
+
# Configurar o texto no PDF
|
63 |
+
c.setFont("Helvetica", 12) # Fonte e tamanho
|
64 |
+
c.drawString(margin_left, margin_top, f"Transcrição do vídeo: {video_filename}") # Título
|
65 |
+
|
66 |
+
# Posição inicial do texto
|
67 |
+
y_position = margin_top - 30
|
68 |
+
|
69 |
+
# Adicionar o texto transcrito ao PDF, quebrando em linhas menores
|
70 |
+
for paragraph in text.split("\n"):
|
71 |
+
while len(paragraph) > 0:
|
72 |
+
# Pegar um trecho que caiba na linha
|
73 |
+
line = paragraph[:line_length]
|
74 |
+
paragraph = paragraph[line_length:]
|
75 |
+
|
76 |
+
# Adicionar a linha ao PDF
|
77 |
+
c.drawString(margin_left, y_position, line.strip())
|
78 |
+
y_position -= line_spacing
|
79 |
+
|
80 |
+
# Se chegar ao final da página, criar uma nova
|
81 |
+
if y_position < 50:
|
82 |
+
c.showPage()
|
83 |
+
c.setFont("Helvetica", 12)
|
84 |
+
y_position = margin_top
|
85 |
+
|
86 |
+
# Salvar o PDF
|
87 |
+
c.save()
|
88 |
+
|
89 |
+
# Excluir o arquivo de áudio temporário
|
90 |
+
os.remove(audio_path)
|
91 |
+
|
92 |
+
# Incrementar o contador para o próximo arquivo
|
93 |
+
contador += 1
|
94 |
+
|
95 |
+
print(f"Transcrições concluídas e salvas como PDFs na pasta '{pdf_root_folder}'.")
|