{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "d8c2d665",
   "metadata": {},
   "source": [
    "# Build a Tiny Packing Helper Agent! (Step by Step) — No API Keys\n",
    "\n",
    "**Who is this for?** Middle‑school explorers.  \n",
    "**What will we do?** Build a tiny *agent* that suggests what to pack for a trip.  \n",
    "**How?** We take small steps. Each code cell does **one thing**. You can **change** values and **re‑run**.\n",
    "\n",
    "> If you can read recipe steps, you can follow this. No machine learning background needed."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "13e81f39",
   "metadata": {},
   "source": [
    "# Setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "da9f4d15",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/opt/miniconda3/envs/cs224n/lib/python3.12/pty.py:95: DeprecationWarning: This process (pid=85331) is multi-threaded, use of forkpty() may lead to deadlocks in the child.\n",
      "  pid, fd = os.forkpty()\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Note: you may need to restart the kernel to use updated packages.\n"
     ]
    }
   ],
   "source": [
    "# 📌 Important: To use the AI \"brain\" (LLM) part of this notebook,\n",
    "# you MUST have Ollama installed on your computer and running in the background.\n",
    "#    1) Download Ollama: https://ollama.com/download\n",
    "#    2) After installing, it will usually auto-run. \n",
    "#       You can check by running this command in your terminal: curl http://localhost:11434/api/tags\n",
    "#    3) Pull a model, e.g.: ollama pull llama2:latest (run this in your terminal)\n",
    "#    4) If Ollama isn’t running, start it from Terminal with: ollama serve\n",
    "#\n",
    "# Without Ollama, the notebook will still run the weather + rules parts,\n",
    "# but the \"AI brain\" step will be skipped or fall back to a smaller local model.\n",
    "\n",
    "PREFER_BACKEND = \"ollama\"         \n",
    "OLLAMA_URL     = \"http://127.0.0.1:11434/api/generate\"\n",
    "OLLAMA_MODEL   = \"llama3.1:8b-instruct\"  # try \"llama3.2:3b-instruct\" if you need smaller/faster\n",
    "\n",
    "# Core Python libraries (all free; no API keys needed)\n",
    "%pip -q install duckduckgo-search requests beautifulsoup4\n",
    "\n",
    "import requests\n",
    "\n",
    "# Map display library\n",
    "try:\n",
    "    import folium\n",
    "except Exception:\n",
    "    %pip -q install folium\n",
    "\n",
    "import os, re, json, math, textwrap, typing as T\n",
    "import requests\n",
    "from bs4 import BeautifulSoup\n",
    "from IPython.display import Markdown, display\n",
    "from duckduckgo_search import DDGS\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7f2c9efe",
   "metadata": {},
   "source": [
    "\n",
    "# Introduction\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0e007fea",
   "metadata": {},
   "source": [
    "When humans pack, we don’t do magic. We often:\n",
    "1) **Check the weather** (how cold? how hot? rain?)  \n",
    "2) **Do a quick search** for local tips (layers? beach? dress codes?)  \n",
    "3) **Use common-sense rules** (hot → light clothes; rain → umbrella)  \n",
    "4) **Make a final list**\n",
    "\n",
    "Today, we'll learn how to make an *agent* which can perform these steps for us!"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9e353940",
   "metadata": {},
   "source": [
    "\n",
    "## 1) What is an *agent*?\n",
    "An **agent** is just a program that:\n",
    "1. Has a **goal** (e.g., make a packing list).  \n",
    "2. Can **use tools** (e.g., a weather website).  \n",
    "3. Repeats a simple loop: **think → act (use a tool) → check → repeat**.  \n",
    "4. **Stops** when done or when a limit is reached.\n",
    "\n",
    "### Our agent’s plan (tools + AI brain)\n",
    "- **Tool:** Find your dream city on a map (Geocoding)\n",
    "- **Tool:** Fetch the **weather** of that city (Open-Meteo: low, high, rain %)  \n",
    "- **Tool:** Do a quick **web search** (DuckDuckGo) for local hints  \n",
    "- **AI brain (Ollama):** Ask a local **LLM** (we'll learn more about this soon) to make a **5-item list** from the coordinates + weather + search results\n",
    "- **Compare & combine:** Our own packing list with the agent's list!\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4638bb4c",
   "metadata": {},
   "source": [
    "## 2) Choose your city and day ✈️\n",
    "Imagine you’re about to take your **DREAM vacation** next week.  \n",
    "Where do you want to go? Paris? Tokyo? San Francisco?  \n",
    "\n",
    "Type your dream **city** in the box below.  \n",
    "Then pick **when** you want to “travel” (today = `0`, tomorrow = `1`, up to `6` days from now).  \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "2bbac9ae",
   "metadata": {
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "City: Paris | Days out: 1\n"
     ]
    }
   ],
   "source": [
    "city = \"Paris\"   # ← try changing this\n",
    "days_out = 1     # ← 0=today, 1=tomorrow, ...\n",
    "print(\"City:\", city, \"| Days out:\", days_out)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3156714e",
   "metadata": {},
   "source": [
    "## 3) Tool #1 — Geocoding (city → coordinates) 🗺️\n",
    "Some weather tools don’t understand names like `\"Paris\"`.  \n",
    "They speak in **numbers**: latitude (north/south) and longitude (east/west).  \n",
    "\n",
    "So we need to **translate** our city name into `(lat, lon)` coordinates.  \n",
    "That’s what **geocoding** does — it’s like looking up your city on a giant world map.\n",
    "\n",
    "**You should see:**  \n",
    "- The exact **latitude/longitude**  \n",
    "- A nice **label** of your city and country  \n",
    "- A map showing where you landed!\n",
    "\n",
    "> Pro-tip: Try changing the city name above and re-run this cell. See how the map jumps!"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "b66c2617",
   "metadata": {
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Coordinates: 48.85341 2.3488 | Label: Paris France\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"width:100%;\"><div style=\"position:relative;width:100%;height:0;padding-bottom:60%;\"><span style=\"color:#565656\">Make this Notebook Trusted to load map: File -> Trust Notebook</span><iframe srcdoc=\"&lt;!DOCTYPE html&gt;\n",
       "&lt;html&gt;\n",
       "&lt;head&gt;\n",
       "    \n",
       "    &lt;meta http-equiv=&quot;content-type&quot; content=&quot;text/html; charset=UTF-8&quot; /&gt;\n",
       "    &lt;script src=&quot;https://cdn.jsdelivr.net/npm/leaflet@1.9.3/dist/leaflet.js&quot;&gt;&lt;/script&gt;\n",
       "    &lt;script src=&quot;https://code.jquery.com/jquery-3.7.1.min.js&quot;&gt;&lt;/script&gt;\n",
       "    &lt;script src=&quot;https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/js/bootstrap.bundle.min.js&quot;&gt;&lt;/script&gt;\n",
       "    &lt;script src=&quot;https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.js&quot;&gt;&lt;/script&gt;\n",
       "    &lt;link rel=&quot;stylesheet&quot; href=&quot;https://cdn.jsdelivr.net/npm/leaflet@1.9.3/dist/leaflet.css&quot;/&gt;\n",
       "    &lt;link rel=&quot;stylesheet&quot; href=&quot;https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/css/bootstrap.min.css&quot;/&gt;\n",
       "    &lt;link rel=&quot;stylesheet&quot; href=&quot;https://netdna.bootstrapcdn.com/bootstrap/3.0.0/css/bootstrap-glyphicons.css&quot;/&gt;\n",
       "    &lt;link rel=&quot;stylesheet&quot; href=&quot;https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@6.2.0/css/all.min.css&quot;/&gt;\n",
       "    &lt;link rel=&quot;stylesheet&quot; href=&quot;https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.css&quot;/&gt;\n",
       "    &lt;link rel=&quot;stylesheet&quot; href=&quot;https://cdn.jsdelivr.net/gh/python-visualization/folium/folium/templates/leaflet.awesome.rotate.min.css&quot;/&gt;\n",
       "    \n",
       "            &lt;meta name=&quot;viewport&quot; content=&quot;width=device-width,\n",
       "                initial-scale=1.0, maximum-scale=1.0, user-scalable=no&quot; /&gt;\n",
       "            &lt;style&gt;\n",
       "                #map_3ae8698309833dd0ecb687f4e2d79475 {\n",
       "                    position: relative;\n",
       "                    width: 100.0%;\n",
       "                    height: 100.0%;\n",
       "                    left: 0.0%;\n",
       "                    top: 0.0%;\n",
       "                }\n",
       "                .leaflet-container { font-size: 1rem; }\n",
       "            &lt;/style&gt;\n",
       "\n",
       "            &lt;style&gt;html, body {\n",
       "                width: 100%;\n",
       "                height: 100%;\n",
       "                margin: 0;\n",
       "                padding: 0;\n",
       "            }\n",
       "            &lt;/style&gt;\n",
       "\n",
       "            &lt;style&gt;#map {\n",
       "                position:absolute;\n",
       "                top:0;\n",
       "                bottom:0;\n",
       "                right:0;\n",
       "                left:0;\n",
       "                }\n",
       "            &lt;/style&gt;\n",
       "\n",
       "            &lt;script&gt;\n",
       "                L_NO_TOUCH = false;\n",
       "                L_DISABLE_3D = false;\n",
       "            &lt;/script&gt;\n",
       "\n",
       "        \n",
       "&lt;/head&gt;\n",
       "&lt;body&gt;\n",
       "    \n",
       "    \n",
       "            &lt;div class=&quot;folium-map&quot; id=&quot;map_3ae8698309833dd0ecb687f4e2d79475&quot; &gt;&lt;/div&gt;\n",
       "        \n",
       "&lt;/body&gt;\n",
       "&lt;script&gt;\n",
       "    \n",
       "    \n",
       "            var map_3ae8698309833dd0ecb687f4e2d79475 = L.map(\n",
       "                &quot;map_3ae8698309833dd0ecb687f4e2d79475&quot;,\n",
       "                {\n",
       "                    center: [48.85341, 2.3488],\n",
       "                    crs: L.CRS.EPSG3857,\n",
       "                    ...{\n",
       "  &quot;zoom&quot;: 6,\n",
       "  &quot;zoomControl&quot;: true,\n",
       "  &quot;preferCanvas&quot;: false,\n",
       "}\n",
       "\n",
       "                }\n",
       "            );\n",
       "\n",
       "            \n",
       "\n",
       "        \n",
       "    \n",
       "            var tile_layer_7f035823f8069134530a444ae277a2ec = L.tileLayer(\n",
       "                &quot;https://tile.openstreetmap.org/{z}/{x}/{y}.png&quot;,\n",
       "                {\n",
       "  &quot;minZoom&quot;: 0,\n",
       "  &quot;maxZoom&quot;: 19,\n",
       "  &quot;maxNativeZoom&quot;: 19,\n",
       "  &quot;noWrap&quot;: false,\n",
       "  &quot;attribution&quot;: &quot;\\u0026copy; \\u003ca href=\\&quot;https://www.openstreetmap.org/copyright\\&quot;\\u003eOpenStreetMap\\u003c/a\\u003e contributors&quot;,\n",
       "  &quot;subdomains&quot;: &quot;abc&quot;,\n",
       "  &quot;detectRetina&quot;: false,\n",
       "  &quot;tms&quot;: false,\n",
       "  &quot;opacity&quot;: 1,\n",
       "}\n",
       "\n",
       "            );\n",
       "        \n",
       "    \n",
       "            tile_layer_7f035823f8069134530a444ae277a2ec.addTo(map_3ae8698309833dd0ecb687f4e2d79475);\n",
       "        \n",
       "    \n",
       "            var marker_bdd0a603b197df2aef9df6d94ad152a7 = L.marker(\n",
       "                [48.85341, 2.3488],\n",
       "                {\n",
       "}\n",
       "            ).addTo(map_3ae8698309833dd0ecb687f4e2d79475);\n",
       "        \n",
       "    \n",
       "            marker_bdd0a603b197df2aef9df6d94ad152a7.bindTooltip(\n",
       "                `&lt;div&gt;\n",
       "                     Paris France\n",
       "                 &lt;/div&gt;`,\n",
       "                {\n",
       "  &quot;sticky&quot;: true,\n",
       "}\n",
       "            );\n",
       "        \n",
       "&lt;/script&gt;\n",
       "&lt;/html&gt;\" style=\"position:absolute;width:100%;height:100%;left:0;top:0;border:none !important;\" allowfullscreen webkitallowfullscreen mozallowfullscreen></iframe></div></div>"
      ],
      "text/plain": [
       "<folium.folium.Map at 0x12fceb5c0>"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "def geocode_city(city_name: str):\n",
    "    \"\"\"\n",
    "    Ask the Open-Meteo geocoding API to find our city.\n",
    "    Returns (lat, lon, label).\n",
    "    \"\"\"\n",
    "    url = \"https://geocoding-api.open-meteo.com/v1/search\"\n",
    "    params = {\"name\": city_name, \"count\": 1, \"language\": \"en\", \"format\": \"json\"}\n",
    "    r = requests.get(url, params=params, timeout=20)  # talk to the API\n",
    "    data = r.json()  # turn the reply into a Python dictionary\n",
    "    results = data.get(\"results\") or []\n",
    "    if not results:\n",
    "        raise ValueError(\"❌ City not found. Try adding a country or nearby big city.\")\n",
    "    first = results[0]\n",
    "    lat = float(first[\"latitude\"])   # north/south\n",
    "    lon = float(first[\"longitude\"])  # east/west\n",
    "    label = f\"{first.get('name','')} {first.get('country','')}\".strip()\n",
    "    return lat, lon, label\n",
    "\n",
    "lat, lon, place_label = geocode_city(city)\n",
    "print(\"Coordinates:\", lat, lon, \"| Label:\", place_label)\n",
    "\n",
    "# Show a mini map\n",
    "m = folium.Map(location=[lat, lon], zoom_start=6)\n",
    "folium.Marker([lat, lon], tooltip=place_label).add_to(m)\n",
    "m"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6e4d577f",
   "metadata": {},
   "source": [
    "## 4) Tool #2 — Weather (the few numbers we care about)\n",
    "\n",
    "When you pack for a trip, the most important things are obviously **snacks** 😄  \n",
    "…but clothes matter too (no one wants to shiver in shorts).\n",
    "\n",
    "**How do we know what clothes to pack?**  \n",
    "We need three clues about your travel day:\n",
    "- **How COLD?** → the **low** temperature (°C)  \n",
    "- **How HOT?** → the **high** temperature (°C)  \n",
    "- **Will it RAIN?** → the **chance of rain** (%)\n",
    "\n",
    "With just these three numbers, we can make a smart packing list.\n",
    "\n",
    "### What’s an API (in plain English)?\n",
    "An **API** is like a friendly robot at a help desk.  \n",
    "You ask it a clear question → it gives you a tidy answer a computer can read.  \n",
    "We’ll ask: “What’s the low temperature, high temperature, and rain chance for this place on this day?”\n",
    "\n",
    "### What is Open-Meteo?\n",
    "**Open-Meteo** is a free weather service with public APIs.  \n",
    "No account needed. You send it **coordinates** (from our geocoding step), and it sends back **forecast data**.\n",
    "\n",
    "> Try it: After you run the code below, change `days_out` or the `city` above and re-run to see different weather!"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "id": "55a6eee7",
   "metadata": {
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Paris France on 2025-09-03: low 17°C, high 23°C, rain chance 68%.\n"
     ]
    }
   ],
   "source": [
    "def get_weather(lat: float, lon: float, day_offset: int = 1):\n",
    "    \"\"\"\n",
    "    Ask Open-Meteo for the forecast and extract:\n",
    "    - date of the forecast day\n",
    "    - low temperature (°C)\n",
    "    - high temperature (°C)\n",
    "    - max chance of rain (%)\n",
    "    \"\"\"\n",
    "    # 1) This is the API “front desk” URL we’re asking\n",
    "    url = \"https://api.open-meteo.com/v1/forecast\"\n",
    "    \n",
    "    # 2) These are our question details (called \"query parameters\")\n",
    "    #    - latitude/longitude: where on Earth?\n",
    "    #    - daily: exactly which numbers we want back\n",
    "    #    - timezone: so the date matches the local place\n",
    "    params = {\n",
    "        \"latitude\": lat,\n",
    "        \"longitude\": lon,\n",
    "        \"daily\": \"temperature_2m_max,temperature_2m_min,precipitation_probability_max\",\n",
    "        \"timezone\": \"auto\",\n",
    "    }\n",
    "    \n",
    "    # 3) Send the request to the API robot and get the JSON (computer-friendly) reply\n",
    "    reply = requests.get(url, params=params, timeout=20).json()\n",
    "    \n",
    "    # 4) Carefully pull out the pieces we need\n",
    "    daily = reply.get(\"daily\", {})\n",
    "    dates = daily.get(\"time\", [])\n",
    "    tmins = daily.get(\"temperature_2m_min\", [])\n",
    "    tmaxs = daily.get(\"temperature_2m_max\", [])\n",
    "    prain = daily.get(\"precipitation_probability_max\", [])\n",
    "    if not dates:\n",
    "        raise ValueError(\"Weather unavailable here right now—try a different city or day.\")\n",
    "    \n",
    "    # 5) Pick the right day (0=today, 1=tomorrow, etc.)\n",
    "    idx = min(day_offset, len(dates) - 1)\n",
    "    \n",
    "    # 6) Return a tiny, easy-to-use package of the numbers we care about\n",
    "    return {\n",
    "        \"date\": dates[idx],\n",
    "        \"tmin_c\": float(tmins[idx]),\n",
    "        \"tmax_c\": float(tmaxs[idx]),\n",
    "        \"p_rain\": int(prain[idx]),\n",
    "    }\n",
    "\n",
    "# Run it! (uses lat/lon from the previous cell and your chosen days_out)\n",
    "wx = get_weather(lat, lon, days_out)\n",
    "\n",
    "# Show a friendly sentence so humans (not just computers) can read it\n",
    "weather_data = f\"{place_label} on {wx['date']}: low {wx['tmin_c']:.0f}°C, high {wx['tmax_c']:.0f}°C, rain chance {wx['p_rain']}%.\"\n",
    "print(weather_data)\n",
    "\n",
    "# Try it:\n",
    "#  - Change 'days_out' above and re-run this cell.\n",
    "#  - Go back and change 'city' (e.g., 'Tokyo, Japan') and re-run the geocoding + this cell."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "abcd82cd",
   "metadata": {},
   "source": [
    "## Pause: Read the numbers — then *you* be the packing agent 🧠\n",
    "\n",
    "Before we let the computer decide, **make your own call** based on the weather:\n",
    "\n",
    "1) **What would *you* pack?**  \n",
    "   Look at the low, high, and rain chance. Write ~5 clothes you’d bring.\n",
    "\n",
    "2) **Units matter.**  \n",
    "   We showed temperatures in **°C**. Do you think your classmates would understand this better in **°C** or **°F**? Why?  \n",
    "   Try both formats below and pick the one (or both!) you’d use if you were designing this app for everyone in class.\n",
    "   \n",
    "2) **Format matters.**  \n",
    "   Do you like having the weather information in one sentence? Would you prefer to have a table instead with information about each day? \n",
    "\n",
    "3) **Bonus Question**  \n",
    "   Why do you think we only allowed vacations up to 6 days from now?🤔 Ask your friends and teacher!\n",
    "\n",
    "> After you write your guess list, we’ll see what the agent suggests and compare!"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "bf35cfb0-e5a3-4c00-9393-1a40c773f84b",
   "metadata": {
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Paris France on 2025-09-03: low 64°F, high 74°F, rain chance 50%.\n"
     ]
    }
   ],
   "source": [
    "def c_to_f(c): \n",
    "    return c * 9/5 + 32\n",
    "\n",
    "print(f\"{place_label} on {wx['date']}: low {c_to_f(wx['tmin_c']):.0f}°F, high {c_to_f(wx['tmax_c']):.0f}°F, rain chance {wx['p_rain']}%.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ed3ad430",
   "metadata": {},
   "source": [
    "\n",
    "## 5) Simple, visible rules → packing items (using *your* brain)\n",
    "Sometimes you don’t need an AI brain at all.  \n",
    "If you know the **inputs** (tempersatures + rain), you can write **your own rules**:\n",
    "\n",
    "- hot → light clothes, sun protection  \n",
    "- rain likely → umbrella / rain jacket  \n",
    "- cool nights → a warmer layer  \n",
    "- always → comfy shoes, charger, water\n",
    "\n",
    "These `if / else` rules are **transparent**: you can see them, edit them, and predict the output.  \n",
    "> Try changing the thresholds to match **your** comfort level and re-run!\n",
    "\n",
    "**Here are our rules (you can change these / add your own!):**\n",
    "- Hot if `high ≥ 29°C` → light clothes, sun protection  \n",
    "- Warm if `22–28°C` → T‑shirt + light layer  \n",
    "- Mild if `15–21°C` → sweater/long sleeves  \n",
    "- Cool if `8–14°C` → jacket  \n",
    "- Cold if `< 8°C` → warm coat  \n",
    "- Rain: if `rain ≥ 60%` → umbrella/rain jacket; if `30–59%` → light rain layer\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "58fb8429",
   "metadata": {
    "trusted": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "• T‑shirt + light layer for evening\n",
      "• Light rain layer (optional)\n",
      "• Comfortable walking shoes\n",
      "• Phone + charger\n",
      "• Water bottle\n"
     ]
    }
   ],
   "source": [
    "\n",
    "def pack_rules(tmin_c: float, tmax_c: float, p_rain: int):\n",
    "    items = []\n",
    "    # ← Try changing the numbers below\n",
    "    if tmax_c >= 29:\n",
    "        items += [\"• Light, breathable clothing\", \"• Sunscreen, sunglasses, hat\"]\n",
    "    elif 22 <= tmax_c < 29:\n",
    "        items += [\"• T‑shirt + light layer for evening\"]\n",
    "    elif 15 <= tmax_c < 22:\n",
    "        items += [\"• Long sleeves or light sweater\"]\n",
    "    elif 8 <= tmax_c < 15:\n",
    "        items += [\"• Jacket or warm sweater\"]\n",
    "    else:\n",
    "        items += [\"• Warm coat; consider gloves/hat\"]\n",
    "\n",
    "    if p_rain >= 60:\n",
    "        items.append(\"• Compact umbrella or rain jacket\")\n",
    "    elif 30 <= p_rain < 60:\n",
    "        items.append(\"• Light rain layer (optional)\")\n",
    "    else:\n",
    "        items.append(\"• No special rain gear needed\")\n",
    "\n",
    "    # Always bring these\n",
    "    items += [\"• Comfortable walking shoes\", \"• Phone + charger\", \"• Water bottle\"]\n",
    "    return items\n",
    "\n",
    "base_list = pack_rules(wx[\"tmin_c\"], wx[\"tmax_c\"], wx[\"p_rain\"])\n",
    "for line in base_list: print(line)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b941b528",
   "metadata": {},
   "source": [
    "## 6) Let’s Look It Up! 🔎\n",
    "\n",
    "When humans plan, we **look things up** to double-check our thinking:\n",
    "- Are there local “gotchas” (dress codes, microclimates, windy evenings)?\n",
    "- Do people recommend **layers**, **umbrellas**, **swimwear**, or **adapters** there?\n",
    "- Does our first guess match what travelers and locals actually say?\n",
    "\n",
    "Our **agent** should do the same. It will search the web, skim a few pages, and turn what it finds into short **hints** that may change the packing list.\n",
    "\n",
    "### What is DuckDuckGo?\n",
    "**DuckDuckGo** is a **privacy-friendly search engine**. We’ll use a tiny Python library to send a query (like “`<city> what to wear packing tips`”) and get back **titles, snippets, and links**—no accounts, no API keys.\n",
    "\n",
    "### What the code below will do\n",
    "1. **Build a search query -- i.e. the question we'll ask DuckDuckGo** (you can change this!).  \n",
    "2. **Search** DuckDuckGo and collect the **top results**.  \n",
    "3. **Open each link** and **extract a few readable sentences** (no AI—just text scraping).  \n",
    "4. Show you the **sources** and **on-page bullets**, then convert them into **agent hints** we’ll feed into our packer.\n",
    "\n",
    "> Reminder: Search results can be messy or outdated—**trust, but verify**. Prefer recent, reputable sources."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e6bb958c",
   "metadata": {
    "trusted": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "💡 Try one of these (or make your own):\n",
      "  • Paris what to wear packing tips\n",
      "  • Paris summer layers vs coat\n",
      "  • Paris walking shoes advice\n",
      "  • Paris dress code restaurants\n",
      "  • Paris rainy season umbrella or jacket\n",
      "  • Paris beach packing list\n",
      "\n",
      "🔎 We searched for:\n",
      "    Paris summer beach what to wear packing tips\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/mt/_z6swqld1xbfdtyyy78gjhhc0000gn/T/ipykernel_85331/599452185.py:35: RuntimeWarning: This package (`duckduckgo_search`) has been renamed to `ddgs`! Use `pip install ddgs` instead.\n",
      "  with DDGS() as ddgs:\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "🌐 Sources (with on-page bullets):\n",
      " 1. Paris - Wikipedia\n",
      "     It is located in the centre of the Île-de-France region. Paris is the fourth-most populous city in the European Union.…\n",
      "     https://en.wikipedia.org/wiki/Paris\n",
      "       • Paris [ a ] is the capital and largest city of France , with an estimated population of 2,048,472 in January 2025 [update] [ 3 ] in an area of more than 105 km 2 (41 sq mi).\n",
      "       • It is located in the centre of the Île-de-France region.\n",
      "       • Paris is the fourth-most populous city in the European Union .\n",
      "\n",
      " 2. THE 15 BEST Things to Do in Paris (2025) - Must-See Attractions\n",
      "     Things to Do in Paris, France: See Tripadvisor's 5,190,822 traveler reviews and photos of Paris tourist attractions.…\n",
      "     https://www.tripadvisor.com/Attractions-g187147-Activities-Paris_Ile_de_France.html\n",
      "      (no readable page bullets)\n",
      "\n",
      " 3. Paris | Definition, Map, Population, Facts, & History | Britannica\n",
      "     Aug 25, 2025 · Paris, city and capital of France, located along the Seine River, in the north-central part of the…\n",
      "     https://www.britannica.com/place/Paris\n",
      "       • Our editors will review what you’ve submitted and determine whether to revise the article.\n",
      "       • Paris is located in the north-central part of France along the Seine River.\n",
      "       • The wind can be sharp and cold in winter and spring.\n",
      "\n",
      "🧭 What we’ll tell our agent based on these sources:\n",
      "    • Bring sunscreen, sunglasses, and a hat\n",
      "    • Plan for layers / a windbreaker\n",
      "    • Consider swimwear and a quick-dry towel\n",
      "    • Pack one smart-casual outfit\n"
     ]
    }
   ],
   "source": [
    "# Things you can change 👇 (try different seasons/activities/extras)\n",
    "SEASON   = \"\"         # e.g., \"summer\", \"winter\", \"rainy season\"\n",
    "ACTIVITY = \"\"         # e.g., \"beach\", \"hike\", \"city walking\"\n",
    "EXTRA    = \"\"         # e.g., \"dress code\", \"windy evenings\"\n",
    "CUSTOM_QUERY = None   # or set to a string to override everything\n",
    "\n",
    "SAFESTARCH = \"strict\"\n",
    "REGION     = \"wt-wt\"  # world-wide; try \"us-en\" for US/English\n",
    "N_RESULTS  = 3        # top N search results to open\n",
    "\n",
    "def build_query(city, season=\"\", activity=\"\", extra=\"\", base=\"what to wear packing tips\"):\n",
    "    \"\"\"Make a readable search like: 'Paris summer beach what to wear packing tips dress code'.\"\"\"\n",
    "    parts = [city, season, activity, base, extra]\n",
    "    q = \" \".join(p for p in parts if p).strip()\n",
    "    return re.sub(r\"\\s+\", \" \", q)\n",
    "\n",
    "def suggest_queries(city):\n",
    "    \"\"\"Show example queries to spark ideas.\"\"\"\n",
    "    samples = [\n",
    "        f\"{city} what to wear packing tips\",\n",
    "        f\"{city} {SEASON or 'summer'} layers vs coat\",\n",
    "        f\"{city} {ACTIVITY or 'walking'} shoes advice\",\n",
    "        f\"{city} dress code restaurants\",\n",
    "        f\"{city} rainy season umbrella or jacket\",\n",
    "        f\"{city} beach packing list\"\n",
    "    ]\n",
    "    print(\"💡 Try one of these (or make your own):\")\n",
    "    for s in samples:\n",
    "        print(\"  •\", s)\n",
    "\n",
    "def search_top(query, n=3, *, region=REGION, safesearch=SAFESTARCH):\n",
    "    \"\"\"Return [{'title','snippet','url'}, ...] using DuckDuckGo.\"\"\"\n",
    "    with DDGS() as ddgs:\n",
    "        results = list(ddgs.text(query, max_results=n, region=region, safesearch=safesearch))\n",
    "    out = []\n",
    "    for r in results:\n",
    "        out.append({\n",
    "            \"title\": (r.get(\"title\") or \"\").strip(),\n",
    "            \"snippet\": (r.get(\"body\") or \"\").strip(),\n",
    "            \"url\": r.get(\"href\") or \"\"\n",
    "        })\n",
    "    return out\n",
    "\n",
    "def fetch_readable_text(url, limit_chars=3000):\n",
    "    \"\"\"Open a link and keep just paragraph text (trimmed).\"\"\"\n",
    "    if not url:\n",
    "        return \"\"\n",
    "    try:\n",
    "        headers = {\"User-Agent\": \"Mozilla/5.0 (Education-Demo)\"}\n",
    "        r = requests.get(url, headers=headers, timeout=15)\n",
    "        r.raise_for_status()\n",
    "        if \"text/html\" not in r.headers.get(\"Content-Type\",\"\"):\n",
    "            return \"\"\n",
    "        soup = BeautifulSoup(r.text, \"html.parser\")\n",
    "        for tag in soup([\"script\",\"style\",\"header\",\"footer\",\"nav\",\"noscript\",\"svg\",\"form\",\"aside\"]):\n",
    "            tag.decompose()\n",
    "        paras = [p.get_text(\" \", strip=True) for p in soup.find_all(\"p\")]\n",
    "        text = re.sub(r\"\\s+\", \" \", \" \".join(paras)).strip()\n",
    "        return text[:limit_chars]\n",
    "    except Exception:\n",
    "        return \"\"\n",
    "\n",
    "def page_bullets(text, k=3):\n",
    "    \"\"\"Pick 2–3 clear sentences as on-page bullets (no AI).\"\"\"\n",
    "    if not text:\n",
    "        return []\n",
    "    sents, bullets = re.split(r\"(?<=[.!?])\\s+\", text), []\n",
    "    for s in sents:\n",
    "        s = s.strip()\n",
    "        if 50 <= len(s) <= 180:\n",
    "            bullets.append(\"• \" + s)\n",
    "        if len(bullets) >= k:\n",
    "            break\n",
    "    return bullets\n",
    "\n",
    "def make_hints(docs, max_hints=8):\n",
    "    \"\"\"Simple keyword rules → tiny packing hints (easy to edit).\"\"\"\n",
    "    blob = \" \".join([(d.get(\"title\",\"\")+\" \"+d.get(\"snippet\",\"\")+\" \"+d.get(\"page_text\",\"\")) for d in docs]).lower()\n",
    "    hints = []\n",
    "    if re.search(r\"\\brain|showers|drizzle|storm|waterproof|monsoon\", blob):\n",
    "        hints.append(\"• Pack a compact umbrella or a light rain jacket\")\n",
    "    if re.search(r\"\\bsun|uv|heatwave|hot|humid|sunscreen|sunglasses\", blob):\n",
    "        hints.append(\"• Bring sunscreen, sunglasses, and a hat\")\n",
    "    if re.search(r\"\\blayer|layering|windy|breezy|changeable|microclimate|chill\", blob):\n",
    "        hints.append(\"• Plan for layers / a windbreaker\")\n",
    "    if re.search(r\"\\bbeach|swim|coast|island|pool|snorkel\", blob):\n",
    "        hints.append(\"• Consider swimwear and a quick-dry towel\")\n",
    "    if re.search(r\"\\bdress code|smart casual|fine dining|nightlife|club\", blob):\n",
    "        hints.append(\"• Pack one smart-casual outfit\")\n",
    "    if re.search(r\"\\bhike|trail|national park|trek|boots\", blob):\n",
    "        hints.append(\"• Sturdy walking/trail shoes\")\n",
    "    if re.search(r\"\\bmosquito|bugs|insect|repellent\", blob):\n",
    "        hints.append(\"• Insect repellent could help\")\n",
    "    if re.search(r\"\\badapter|plug|voltage|type [a-z]\\b\", blob):\n",
    "        hints.append(\"• Travel power adapter (check plug type)\")\n",
    "    # remove duplicates, preserve ordering\n",
    "    seen, clean = set(), []\n",
    "    for h in hints:\n",
    "        if h not in seen:\n",
    "            seen.add(h); clean.append(h)\n",
    "    return clean[:max_hints]\n",
    "\n",
    "def lookup_tips_for(city, n_results=N_RESULTS, *, custom_query=CUSTOM_QUERY, season=SEASON, activity=ACTIVITY, extra=EXTRA, show_examples=True):\n",
    "    \"\"\"\n",
    "    One call does it all:\n",
    "      1) Build (or use) a query\n",
    "      2) Search DuckDuckGo\n",
    "      3) Open each link and grab readable text\n",
    "      4) Make on-page bullets (no AI)\n",
    "      5) Turn everything into short agent hints\n",
    "      6) Print a kid-friendly summary\n",
    "    Returns: (docs, hints)\n",
    "    \"\"\"\n",
    "    if show_examples:\n",
    "        suggest_queries(city)\n",
    "\n",
    "    query = custom_query or build_query(city, season=season, activity=activity, extra=extra)\n",
    "    print(\"\\n🔎 We searched for:\\n   \", query)\n",
    "\n",
    "    hits = search_top(query, n=n_results)\n",
    "    docs = []\n",
    "    for h in hits:\n",
    "        text = fetch_readable_text(h[\"url\"], limit_chars=3000)\n",
    "        bullets = page_bullets(text, k=3)\n",
    "        docs.append({\"title\": h[\"title\"], \"snippet\": h[\"snippet\"], \"url\": h[\"url\"],\n",
    "                     \"page_text\": text, \"page_bullets\": bullets})\n",
    "\n",
    "    print(\"\\n🌐 Sources (with on-page bullets):\")\n",
    "    if not docs:\n",
    "        print(\"   (No results — try a broader query or another city.)\")\n",
    "    for i, d in enumerate(docs, 1):\n",
    "        print(f\" {i}. {d['title'] or '(no title)'}\")\n",
    "        if d[\"snippet\"]:\n",
    "            print(\"    \", textwrap.shorten(d[\"snippet\"], width=120, placeholder=\"…\"))\n",
    "        if d[\"url\"]:\n",
    "            print(\"    \", d[\"url\"])\n",
    "        if d[\"page_bullets\"]:\n",
    "            for b in d[\"page_bullets\"]:\n",
    "                print(\"      \", b)\n",
    "        else:\n",
    "            print(\"      (no readable page bullets)\")\n",
    "        print()\n",
    "\n",
    "    hints = make_hints(docs, max_hints=8)\n",
    "\n",
    "    print(\"🧭 What we’ll tell our agent based on these sources:\")\n",
    "    if not hints:\n",
    "        print(\"   (No obvious hints found — the agent will rely on weather + your rules.)\")\n",
    "    for h in hints:\n",
    "        print(\"   \", h)\n",
    "\n",
    "    return docs, hints\n",
    "\n",
    "# 👉 Try it:\n",
    "docs, agent_hints = lookup_tips_for(city, season=\"summer\", activity=\"beach\")\n",
    "# docs, agent_hints = lookup_tips_for(city, custom_query=f\"{city} rainy season what to wear\", show_examples=False)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f038a114",
   "metadata": {},
   "source": [
    "## Let an LLM make **its own** 5-item packing list 🧠➡️🧳\n",
    "\n",
    "So far, *you* acted like an agent:  \n",
    "- You looked at the **numbers** (weather forecast).  \n",
    "- You did a quick **search** for local tips.  \n",
    "- You applied your own **rules** to decide what to pack.  \n",
    "\n",
    "Now let’s hand the same job to a **computer brain**.\n",
    "\n",
    "### What is an LLM?\n",
    "**LLM** stands for **Large Language Model**.  \n",
    "It’s a kind of AI program trained on a huge collection of text (books, websites, articles).  \n",
    "Because of that training, it has learned patterns of how people write and can **generate new sentences** that *sound like* a human wrote them.\n",
    "\n",
    "Think of it like this:\n",
    "- Imagine you read millions of travel blogs, conversations, and packing lists.  \n",
    "- Later, when someone asks, “What should I pack for Paris if it’s 10°C, raining, and locals suggest layers?”, you could guess a good answer.  \n",
    "That’s what an LLM does — only much faster.\n",
    "\n",
    "### Our plan\n",
    "1. We’ll give the LLM the **same inputs you saw**:\n",
    "   - Your city’s **location** (coordinates).  \n",
    "   - The **weather forecast** (low, high, rain %).  \n",
    "   - The **web search hints** (from DuckDuckGo).  \n",
    "2. We’ll ask it: *“Write exactly 5 packing items for this trip.”*  \n",
    "3. We’ll compare its 5 items to your human list and our simple rule-based list."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "id": "5b7faef4",
   "metadata": {
    "trusted": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "🧾 Prompt sent to the AI:\n",
      "\n",
      "You are a helpful packing assistant for kids.\n",
      "\n",
      "Use the trip info below to choose EXACTLY FIVE (5) short packing items.\n",
      "\n",
      "RULES:\n",
      "- Output exactly 5 bullet lines, nothing else.\n",
      "- Start each line with the bullet \"• \" (dot + space).\n",
      "- No numbers. No extra sentences before or after. No explanations.\n",
      "- Use the weather and local tips when deciding.\n",
      "\n",
      "TRIP INFO:\n",
      "City: Paris France\n",
      "Coordinates: 48.8534, 2.3488\n",
      "Weather: Paris France on 2025-09-03: low 17°C, high 23°C, rain chance 68%.\n",
      "\n",
      "Local tips from the web:\n",
      "- Bring sunscreen, sunglasses, and a hat\n",
      "- Plan for layers / a windbreaker\n",
      "- Consider swimwear and a quick-dry towel\n",
      "- Pack one smart-casual outfit\n",
      "\n",
      "🤖 AI's 5-item packing list:\n",
      "  • Sunscreen\n",
      "  • Sunglasses\n",
      "  • Hat\n",
      "  • Swimsuit\n",
      "  • Quick-dry towel\n"
     ]
    }
   ],
   "source": [
    "import requests\n",
    "\n",
    "OLLAMA_URL   = \"http://127.0.0.1:11434\"\n",
    "OLLAMA_MODEL = \"llama2:latest\"   # ← match this with the model you `pull`ed\n",
    "\n",
    "def _require_model(model: str):\n",
    "    r = requests.get(f\"{OLLAMA_URL}/api/tags\", timeout=10)\n",
    "    r.raise_for_status()\n",
    "    names = [m.get(\"name\",\"\") for m in (r.json().get(\"models\") or [])]\n",
    "    if model not in names:\n",
    "        raise RuntimeError(\n",
    "            f\"Model '{model}' not installed. Installed: {names or '(none)'}\\n\"\n",
    "            f\"Fix: run `ollama pull {model}` in Terminal.\"\n",
    "        )\n",
    "\n",
    "def llm_make_5_items_with_hints(\n",
    "    lat: float,\n",
    "    lon: float,\n",
    "    city_label: str,\n",
    "    weather_data: str,          # ← your existing weather sentence string\n",
    "    agent_hints: list[str],     # ← from your DuckDuckGo step\n",
    "    model: str = OLLAMA_MODEL,\n",
    "    endpoint: str = f\"{OLLAMA_URL}/api/generate\",\n",
    "    show_prompt: bool = True\n",
    "):\n",
    "    _require_model(model)\n",
    "\n",
    "    # Turn web hints into plain dashes for the prompt\n",
    "    hints_block = \"\\n\".join(f\"- {h.lstrip('• ').strip()}\" for h in (agent_hints or [])) \\\n",
    "                  or \"- (no special local tips found)\"\n",
    "\n",
    "    prompt = f\"\"\"\n",
    "You are a helpful packing assistant for kids.\n",
    "\n",
    "Use the trip info below to choose EXACTLY FIVE (5) short packing items.\n",
    "\n",
    "RULES:\n",
    "- Output exactly 5 bullet lines, nothing else.\n",
    "- Start each line with the bullet \"• \" (dot + space).\n",
    "- No numbers. No extra sentences before or after. No explanations.\n",
    "- Use the weather and local tips when deciding.\n",
    "\n",
    "TRIP INFO:\n",
    "City: {city_label}\n",
    "Coordinates: {lat:.4f}, {lon:.4f}\n",
    "Weather: {weather_data}\n",
    "\n",
    "Local tips from the web:\n",
    "{hints_block}\n",
    "\"\"\".strip()\n",
    "\n",
    "    if show_prompt:\n",
    "        print(\"🧾 Prompt sent to the AI:\\n\")\n",
    "        print(prompt)\n",
    "\n",
    "    resp = requests.post(\n",
    "        endpoint,\n",
    "        json={\"model\": model, \"prompt\": prompt, \"stream\": False},\n",
    "        timeout=90\n",
    "    )\n",
    "    # If this raises, you'll see the server's message (e.g., model not found)\n",
    "    resp.raise_for_status()\n",
    "    raw = (resp.json().get(\"response\") or \"\").strip()\n",
    "\n",
    "    # Normalize: force bullets, keep at most 5\n",
    "    lines = [ln.strip() for ln in raw.splitlines() if ln.strip()]\n",
    "    bullets = []\n",
    "    for ln in lines:\n",
    "        ln = ln.lstrip(\"-*•\").lstrip(\"0123456789. \").strip()\n",
    "        if ln:\n",
    "            bullets.append(\"• \" + ln)\n",
    "        if len(bullets) == 5:\n",
    "            break\n",
    "\n",
    "    print(\"\\n🤖 AI's 5-item packing list:\")\n",
    "    for b in bullets:\n",
    "        print(\" \", b)\n",
    "\n",
    "    return \"\\n\".join(bullets), \"ollama\"\n",
    "\n",
    "# ✅ Example call (you already have `weather_data` as the sentence)\n",
    "llm_text, backend = llm_make_5_items_with_hints(lat, lon, place_label, weather_data, agent_hints, model=\"llama2:latest\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b3ca20a2",
   "metadata": {},
   "source": [
    "## 8) Recap & Reflection — did our agent think like you?\n",
    "\n",
    "**What the agent actually did (end-to-end):**\n",
    "1) **Geocoded** your city → got coordinates (numbers computers use).  \n",
    "2) **Fetched weather** from Open-Meteo → low / high / rain %.  \n",
    "3) **Looked it up** with DuckDuckGo → skimmed pages and pulled quick on-page bullets.  \n",
    "4) **Asked an AI brain (Ollama LLM)** with the **same info you saw** (city, coords, weather sentence, web hints) to make **exactly 5** items.  \n",
    "\n",
    "**Agent loop:** think → act (use a tool) → check → *repeat* → stop.\n",
    "\n",
    "---\n",
    "\n",
    "### Reflect (write short answers)\n",
    "- **Compare lists:** What did *you* include that the **agent** missed? What did the agent add that you didn’t consider?  \n",
    "- **Search impact:** Which web hint actually changed the packing list? Any hint that felt wrong or outdated?  \n",
    "- **Rules vs. LLM:** If the AI’s 5 items feel off, what **extra info** could we give it to improve? \n",
    "- **Trust check:** Can you blindly trust the AI? What is your **verification plan** (source recency, official sites)?  \n",
    "- **Speed vs. effort:** Which was faster for you— reading weather + a couple snippets, or running the agent?\n",
    "\n",
    "---\n",
    "\n",
    "### You now have a **packing helper**\n",
    "Next time you travel, change the **city** and **days_out**, hit **Run**, and you’ve got a fresh, explainable packing list—built from **weather**, **web tips**, **your rules**, and an **AI brain**.  \n",
    "Pro tip: still sanity-check sources and tweak the rules to fit *you*.\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "cs224n",
   "language": "python",
   "name": "cs224n"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
