File size: 13,044 Bytes
026b128
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1">
  <title>Shami's Agent System</title>
  <style>
    :root {
      --bg: #0a0a0a; --surface: #111; --surface2: #1a1a1a;
      --border: #222; --text: #e0e0e0; --text-dim: #666;
      --accent: #4a9eff; --accent-dim: #1a3a5c;
      --green: #4ade80; --yellow: #fbbf24; --red: #f87171;
    }
    * { margin: 0; padding: 0; box-sizing: border-box; }
    body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', system-ui, sans-serif; background: var(--bg); color: var(--text); height: 100vh; display: flex; }

    /* Sidebar */
    .sidebar { width: 260px; background: var(--surface); border-right: 1px solid var(--border); display: flex; flex-direction: column; flex-shrink: 0; }
    .sidebar-header { padding: 20px; border-bottom: 1px solid var(--border); }
    .sidebar-header h1 { font-size: 16px; color: #fff; letter-spacing: -0.3px; }
    .sidebar-header p { font-size: 11px; color: var(--text-dim); margin-top: 4px; }
    .agent-list { flex: 1; overflow-y: auto; padding: 8px; }
    .agent-btn { width: 100%; text-align: left; padding: 12px 14px; background: none; border: 1px solid transparent; border-radius: 8px; color: var(--text); cursor: pointer; margin-bottom: 4px; font-size: 13px; display: flex; align-items: center; gap: 10px; transition: all 0.15s; }
    .agent-btn:hover { background: var(--surface2); border-color: var(--border); }
    .agent-btn.active { background: var(--accent-dim); border-color: var(--accent); color: #fff; }
    .agent-btn .icon { font-size: 18px; width: 28px; text-align: center; }
    .agent-btn .info { flex: 1; }
    .agent-btn .name { font-weight: 600; display: block; }
    .agent-btn .desc { font-size: 11px; color: var(--text-dim); display: block; margin-top: 2px; }
    .agent-btn.active .desc { color: #aaa; }
    .sidebar-footer { padding: 12px 16px; border-top: 1px solid var(--border); font-size: 11px; color: var(--text-dim); }
    .sidebar-footer a { color: var(--accent); text-decoration: none; }

    /* Main area */
    .main { flex: 1; display: flex; flex-direction: column; min-width: 0; }
    .main-header { padding: 14px 24px; border-bottom: 1px solid var(--border); background: var(--surface); display: flex; align-items: center; justify-content: space-between; }
    .main-header h2 { font-size: 15px; color: #fff; }
    .main-header .tools { font-size: 11px; color: var(--text-dim); }
    .main-header .tools span { background: var(--surface2); padding: 2px 8px; border-radius: 4px; margin-left: 4px; border: 1px solid var(--border); }

    /* Chat area */
    .chat { flex: 1; overflow-y: auto; padding: 24px; display: flex; flex-direction: column; gap: 16px; }
    .msg { max-width: 85%; padding: 14px 18px; border-radius: 12px; line-height: 1.6; font-size: 14px; }
    .msg.user { align-self: flex-end; background: var(--accent-dim); border: 1px solid #2a4a6c; }
    .msg.assistant { align-self: flex-start; background: var(--surface2); border: 1px solid var(--border); }
    .msg pre { background: var(--bg); padding: 12px; border-radius: 8px; overflow-x: auto; margin: 10px 0; font-size: 13px; border: 1px solid var(--border); }
    .msg code { font-family: 'SF Mono', 'Fira Code', monospace; font-size: 13px; }
    .msg p code { background: var(--bg); padding: 2px 6px; border-radius: 4px; }
    .msg table { border-collapse: collapse; margin: 10px 0; width: 100%; }
    .msg th, .msg td { border: 1px solid var(--border); padding: 6px 10px; text-align: left; font-size: 13px; }
    .msg th { background: var(--bg); }
    .msg h1, .msg h2, .msg h3 { margin: 12px 0 6px; color: #fff; }
    .msg h1 { font-size: 18px; } .msg h2 { font-size: 16px; } .msg h3 { font-size: 14px; }
    .msg ul, .msg ol { padding-left: 20px; margin: 6px 0; }
    .msg a { color: var(--accent); }
    .msg strong { color: #fff; }

    /* Tool call indicator */
    .tool-indicator { align-self: flex-start; font-size: 12px; color: var(--text-dim); padding: 6px 12px; background: var(--surface); border: 1px solid var(--border); border-radius: 6px; display: flex; align-items: center; gap: 6px; }
    .tool-indicator .dot { width: 6px; height: 6px; border-radius: 50%; background: var(--green); animation: pulse 1s infinite; }
    @keyframes pulse { 0%,100% { opacity: 1; } 50% { opacity: 0.4; } }

    /* Meta info after response */
    .meta { align-self: flex-start; font-size: 11px; color: var(--text-dim); padding: 4px 0; }

    /* Input */
    .input-area { padding: 16px 24px; border-top: 1px solid var(--border); background: var(--surface); display: flex; gap: 12px; }
    .input-area textarea { flex: 1; padding: 12px 16px; background: var(--surface2); border: 1px solid var(--border); border-radius: 8px; color: var(--text); font-size: 14px; font-family: inherit; outline: none; resize: none; min-height: 44px; max-height: 120px; }
    .input-area textarea:focus { border-color: var(--accent); }
    .input-area button { padding: 12px 24px; background: var(--accent); color: #fff; border: none; border-radius: 8px; font-size: 14px; cursor: pointer; font-weight: 500; align-self: flex-end; }
    .input-area button:hover { background: #3a8eef; }
    .input-area button:disabled { background: #333; cursor: not-allowed; }

    /* Welcome */
    .welcome { flex: 1; display: flex; flex-direction: column; align-items: center; justify-content: center; text-align: center; padding: 40px; }
    .welcome h2 { font-size: 22px; margin-bottom: 8px; color: #fff; }
    .welcome p { color: var(--text-dim); font-size: 14px; max-width: 400px; line-height: 1.6; }
    .examples { margin-top: 24px; display: flex; flex-direction: column; gap: 8px; }
    .example-btn { padding: 10px 16px; background: var(--surface2); border: 1px solid var(--border); border-radius: 8px; color: var(--text); cursor: pointer; font-size: 13px; text-align: left; transition: all 0.15s; }
    .example-btn:hover { border-color: var(--accent); background: var(--accent-dim); }

    @media (max-width: 768px) {
      body { flex-direction: column; }
      .sidebar { width: 100%; height: auto; max-height: 40vh; border-right: none; border-bottom: 1px solid var(--border); }
    }
  </style>
</head>
<body>
  <div class="sidebar">
    <div class="sidebar-header">
      <h1>Shami's Agents</h1>
      <p>AI-powered tools for real work</p>
    </div>
    <div class="agent-list" id="agentList"></div>
    <div class="sidebar-footer">
      Groq + Tavily &middot; <a href="/health" target="_blank">API Health</a> &middot; <a href="/docs" target="_blank">API Docs</a>
    </div>
  </div>

  <div class="main">
    <div class="main-header" id="mainHeader">
      <h2 id="headerTitle">Select an Agent</h2>
      <div class="tools" id="headerTools"></div>
    </div>
    <div class="chat" id="chat">
      <div class="welcome" id="welcome">
        <h2>What do you need done?</h2>
        <p>Pick an agent from the sidebar, or try an example below.</p>
        <div class="examples" id="examples"></div>
      </div>
    </div>
    <div class="input-area">
      <textarea id="input" placeholder="Give the agent a goal..." rows="1"></textarea>
      <button id="send" onclick="sendMessage()">Run</button>
    </div>
  </div>

  <script>
    let currentAgent = null;
    let agents = {};

    const EXAMPLES = {
      search: "Find 5 remote AI/LLM engineer jobs paying $5K+/month, Europe or Gulf region",
      research: "Compare Supabase vs Firebase vs PlanetScale for a SaaS with 10K users",
      review: "Review this PR: https://github.com/vercel/next.js/pull/78498",
      upwork: "Draft a proposal for this job: I need a developer to build a custom AI chatbot for my e-commerce store that can answer product questions using our catalog",
      n8n: "Hi, I found your profile on Upwork. We need someone to build an AI agent that processes customer support tickets automatically. Budget is $2000-3000. Are you available?",
    };

    // Load agents
    fetch("/agents").then(r => r.json()).then(data => {
      agents = {};
      const list = document.getElementById("agentList");
      const examples = document.getElementById("examples");
      data.agents.forEach(a => {
        agents[a.id] = a;
        const btn = document.createElement("button");
        btn.className = "agent-btn";
        btn.dataset.id = a.id;
        btn.innerHTML = `<span class="icon">${a.icon}</span><span class="info"><span class="name">${a.name}</span><span class="desc">${a.description}</span></span>`;
        btn.onclick = () => selectAgent(a.id);
        list.appendChild(btn);

        if (EXAMPLES[a.id]) {
          const ex = document.createElement("button");
          ex.className = "example-btn";
          ex.textContent = `${a.icon} ${EXAMPLES[a.id].slice(0, 60)}...`;
          ex.onclick = () => { selectAgent(a.id); document.getElementById("input").value = EXAMPLES[a.id]; sendMessage(); };
          examples.appendChild(ex);
        }
      });
    });

    function selectAgent(id) {
      currentAgent = id;
      const a = agents[id];
      document.querySelectorAll(".agent-btn").forEach(b => b.classList.toggle("active", b.dataset.id === id));
      document.getElementById("headerTitle").textContent = `${a.icon} ${a.name}`;
      document.getElementById("headerTools").innerHTML = a.tools.map(t => `<span>${t}</span>`).join("");
      document.getElementById("welcome")?.remove();
      document.getElementById("input").placeholder = `Goal for ${a.name}...`;
      document.getElementById("input").focus();
    }

    const input = document.getElementById("input");
    input.addEventListener("keydown", e => { if (e.key === "Enter" && !e.shiftKey) { e.preventDefault(); sendMessage(); } });
    input.addEventListener("input", () => { input.style.height = "auto"; input.style.height = Math.min(input.scrollHeight, 120) + "px"; });

    async function sendMessage() {
      const msg = input.value.trim();
      if (!msg || !currentAgent) { if (!currentAgent) alert("Pick an agent first"); return; }
      input.value = "";
      input.style.height = "auto";
      const btn = document.getElementById("send");
      btn.disabled = true;
      const chat = document.getElementById("chat");

      chat.innerHTML += `<div class="msg user">${esc(msg)}</div>`;
      const toolDiv = document.createElement("div");
      toolDiv.className = "tool-indicator";
      toolDiv.innerHTML = `<span class="dot"></span> ${agents[currentAgent].name} working...`;
      chat.appendChild(toolDiv);
      chat.scrollTop = chat.scrollHeight;

      const start = Date.now();
      try {
        const res = await fetch(`/agent/${currentAgent}`, {
          method: "POST",
          headers: { "Content-Type": "application/json" },
          body: JSON.stringify({ message: msg, conversation_id: `web-${currentAgent}-${Date.now()}` }),
        });
        const data = await res.json();
        toolDiv.remove();

        if (!res.ok) {
          chat.innerHTML += `<div class="msg assistant" style="border-color:var(--red)">Error: ${esc(data.detail || JSON.stringify(data))}</div>`;
        } else {
          chat.innerHTML += `<div class="msg assistant">${renderMd(data.result)}</div>`;
          const elapsed = ((Date.now() - start) / 1000).toFixed(1);
          const toolsInfo = data.tools_used.length > 0 ? ` &middot; Tools: ${data.tools_used.join(", ")}` : "";
          chat.innerHTML += `<div class="meta">${data.tool_calls_made} tool calls &middot; ${data.iterations} iterations &middot; ${elapsed}s${toolsInfo}</div>`;
        }
      } catch (e) {
        toolDiv.remove();
        chat.innerHTML += `<div class="msg assistant" style="border-color:var(--red)">Connection error: ${esc(e.message)}</div>`;
      }
      chat.scrollTop = chat.scrollHeight;
      btn.disabled = false;
      input.focus();
    }

    function esc(s) { return s.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;"); }

    function renderMd(s) {
      // Basic markdown rendering
      let html = esc(s);
      // Code blocks
      html = html.replace(/```(\w*)\n([\s\S]*?)```/g, (_, lang, code) => `<pre><code>${code}</code></pre>`);
      // Inline code
      html = html.replace(/`([^`]+)`/g, "<code>$1</code>");
      // Headers
      html = html.replace(/^### (.+)$/gm, "<h3>$1</h3>");
      html = html.replace(/^## (.+)$/gm, "<h2>$1</h2>");
      html = html.replace(/^# (.+)$/gm, "<h1>$1</h1>");
      // Bold
      html = html.replace(/\*\*([^*]+)\*\*/g, "<strong>$1</strong>");
      // Italic
      html = html.replace(/\*([^*]+)\*/g, "<em>$1</em>");
      // Links
      html = html.replace(/\[([^\]]+)\]\(([^)]+)\)/g, '<a href="$2" target="_blank">$1</a>');
      // Lists
      html = html.replace(/^- (.+)$/gm, "<li>$1</li>");
      html = html.replace(/(<li>.*<\/li>\n?)+/g, m => `<ul>${m}</ul>`);
      // Line breaks
      html = html.replace(/\n\n/g, "</p><p>");
      html = html.replace(/\n/g, "<br>");
      return `<p>${html}</p>`;
    }
  </script>
</body>
</html>