@agenticmail/enterprise 0.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/ARCHITECTURE.md +183 -0
- package/agenticmail-enterprise.db +0 -0
- package/dashboards/README.md +120 -0
- package/dashboards/dotnet/Program.cs +261 -0
- package/dashboards/express/app.js +146 -0
- package/dashboards/go/main.go +513 -0
- package/dashboards/html/index.html +535 -0
- package/dashboards/java/AgenticMailDashboard.java +376 -0
- package/dashboards/php/index.php +414 -0
- package/dashboards/python/app.py +273 -0
- package/dashboards/ruby/app.rb +195 -0
- package/dist/chunk-77IDQJL3.js +7 -0
- package/dist/chunk-7RGCCHIT.js +115 -0
- package/dist/chunk-DXNKR3TG.js +1355 -0
- package/dist/chunk-IQWA44WT.js +970 -0
- package/dist/chunk-LCUZGIDH.js +965 -0
- package/dist/chunk-N2JVTNNJ.js +2553 -0
- package/dist/chunk-O462UJBH.js +363 -0
- package/dist/chunk-PNKVD2UK.js +26 -0
- package/dist/cli.js +218 -0
- package/dist/dashboard/index.html +558 -0
- package/dist/db-adapter-DEWEFNIV.js +7 -0
- package/dist/dynamodb-CCGL2E77.js +426 -0
- package/dist/engine/index.js +1261 -0
- package/dist/index.js +522 -0
- package/dist/mongodb-ODTXIVPV.js +319 -0
- package/dist/mysql-RM3S2FV5.js +521 -0
- package/dist/postgres-LN7A6MGQ.js +518 -0
- package/dist/routes-2JEPIIKC.js +441 -0
- package/dist/routes-74ZLKJKP.js +399 -0
- package/dist/server.js +7 -0
- package/dist/sqlite-3K5YOZ4K.js +439 -0
- package/dist/turso-LDWODSDI.js +442 -0
- package/package.json +49 -0
- package/src/admin/routes.ts +331 -0
- package/src/auth/routes.ts +130 -0
- package/src/cli.ts +260 -0
- package/src/dashboard/index.html +558 -0
- package/src/db/adapter.ts +230 -0
- package/src/db/dynamodb.ts +456 -0
- package/src/db/factory.ts +51 -0
- package/src/db/mongodb.ts +360 -0
- package/src/db/mysql.ts +472 -0
- package/src/db/postgres.ts +479 -0
- package/src/db/sql-schema.ts +123 -0
- package/src/db/sqlite.ts +391 -0
- package/src/db/turso.ts +411 -0
- package/src/deploy/fly.ts +368 -0
- package/src/deploy/managed.ts +213 -0
- package/src/engine/activity.ts +474 -0
- package/src/engine/agent-config.ts +429 -0
- package/src/engine/agenticmail-bridge.ts +296 -0
- package/src/engine/approvals.ts +278 -0
- package/src/engine/db-adapter.ts +682 -0
- package/src/engine/db-schema.ts +335 -0
- package/src/engine/deployer.ts +595 -0
- package/src/engine/index.ts +134 -0
- package/src/engine/knowledge.ts +486 -0
- package/src/engine/lifecycle.ts +635 -0
- package/src/engine/openclaw-hook.ts +371 -0
- package/src/engine/routes.ts +528 -0
- package/src/engine/skills.ts +473 -0
- package/src/engine/tenant.ts +345 -0
- package/src/engine/tool-catalog.ts +189 -0
- package/src/index.ts +64 -0
- package/src/lib/resilience.ts +326 -0
- package/src/middleware/index.ts +286 -0
- package/src/server.ts +310 -0
- package/tsconfig.json +14 -0
|
@@ -0,0 +1,376 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AgenticMail Enterprise Dashboard โ Java Edition
|
|
3
|
+
*
|
|
4
|
+
* ZERO dependencies. Uses only JDK built-in classes (Java 11+).
|
|
5
|
+
* No Spring, no Maven, no Gradle needed.
|
|
6
|
+
*
|
|
7
|
+
* Setup:
|
|
8
|
+
* javac AgenticMailDashboard.java
|
|
9
|
+
* java AgenticMailDashboard
|
|
10
|
+
*
|
|
11
|
+
* Or with env var:
|
|
12
|
+
* AGENTICMAIL_URL=https://your-company.agenticmail.cloud java AgenticMailDashboard
|
|
13
|
+
*
|
|
14
|
+
* Single-file, compiles and runs directly.
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
import com.sun.net.httpserver.*;
|
|
18
|
+
import java.io.*;
|
|
19
|
+
import java.net.*;
|
|
20
|
+
import java.net.http.*;
|
|
21
|
+
import java.net.http.HttpRequest;
|
|
22
|
+
import java.net.http.HttpResponse;
|
|
23
|
+
import java.nio.charset.StandardCharsets;
|
|
24
|
+
import java.util.*;
|
|
25
|
+
import java.util.concurrent.*;
|
|
26
|
+
import java.util.regex.*;
|
|
27
|
+
import java.util.stream.Collectors;
|
|
28
|
+
|
|
29
|
+
public class AgenticMailDashboard {
|
|
30
|
+
|
|
31
|
+
static String API_URL = "http://localhost:3000";
|
|
32
|
+
static final Map<String, Map<String, Object>> sessions = new ConcurrentHashMap<>();
|
|
33
|
+
static final java.net.http.HttpClient client = java.net.http.HttpClient.newBuilder()
|
|
34
|
+
.connectTimeout(java.time.Duration.ofSeconds(10)).build();
|
|
35
|
+
|
|
36
|
+
// โโโ API Client โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
|
|
37
|
+
|
|
38
|
+
@SuppressWarnings("unchecked")
|
|
39
|
+
static Map<String, Object> api(String path, String method, String token, String body) {
|
|
40
|
+
try {
|
|
41
|
+
var builder = HttpRequest.newBuilder()
|
|
42
|
+
.uri(URI.create(API_URL + path))
|
|
43
|
+
.header("Content-Type", "application/json")
|
|
44
|
+
.timeout(java.time.Duration.ofSeconds(10));
|
|
45
|
+
if (token != null) builder.header("Authorization", "Bearer " + token);
|
|
46
|
+
|
|
47
|
+
switch (method) {
|
|
48
|
+
case "POST": builder.POST(HttpRequest.BodyPublishers.ofString(body != null ? body : "")); break;
|
|
49
|
+
case "PATCH": builder.method("PATCH", HttpRequest.BodyPublishers.ofString(body != null ? body : "")); break;
|
|
50
|
+
case "DELETE": builder.DELETE(); break;
|
|
51
|
+
default: builder.GET();
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
var resp = client.send(builder.build(), HttpResponse.BodyHandlers.ofString());
|
|
55
|
+
return parseJson(resp.body());
|
|
56
|
+
} catch (Exception e) {
|
|
57
|
+
return Map.of("error", e.getMessage());
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// โโโ Minimal JSON Parser (no dependencies) โโโโโโโโโโ
|
|
62
|
+
|
|
63
|
+
@SuppressWarnings("unchecked")
|
|
64
|
+
static Map<String, Object> parseJson(String json) {
|
|
65
|
+
// Simple recursive descent parser for JSON objects
|
|
66
|
+
try {
|
|
67
|
+
return (Map<String, Object>) new JsonParser(json.trim()).parseValue();
|
|
68
|
+
} catch (Exception e) {
|
|
69
|
+
return Map.of("error", "Parse error: " + e.getMessage());
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
static String toJson(Map<String, String> map) {
|
|
74
|
+
return "{" + map.entrySet().stream()
|
|
75
|
+
.map(e -> "\"" + e.getKey() + "\":\"" + e.getValue().replace("\"", "\\\"") + "\"")
|
|
76
|
+
.collect(Collectors.joining(",")) + "}";
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// Minimal JSON parser
|
|
80
|
+
static class JsonParser {
|
|
81
|
+
String s; int pos;
|
|
82
|
+
JsonParser(String s) { this.s = s; this.pos = 0; }
|
|
83
|
+
|
|
84
|
+
char peek() { skipWs(); return pos < s.length() ? s.charAt(pos) : 0; }
|
|
85
|
+
char next() { skipWs(); return pos < s.length() ? s.charAt(pos++) : 0; }
|
|
86
|
+
void skipWs() { while (pos < s.length() && " \t\n\r".indexOf(s.charAt(pos)) >= 0) pos++; }
|
|
87
|
+
|
|
88
|
+
Object parseValue() {
|
|
89
|
+
char c = peek();
|
|
90
|
+
if (c == '{') return parseObject();
|
|
91
|
+
if (c == '[') return parseArray();
|
|
92
|
+
if (c == '"') return parseString();
|
|
93
|
+
if (c == 't') { pos += 4; return true; }
|
|
94
|
+
if (c == 'f') { pos += 5; return false; }
|
|
95
|
+
if (c == 'n') { pos += 4; return null; }
|
|
96
|
+
return parseNumber();
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
Map<String, Object> parseObject() {
|
|
100
|
+
Map<String, Object> map = new LinkedHashMap<>();
|
|
101
|
+
next(); // {
|
|
102
|
+
while (peek() != '}') {
|
|
103
|
+
String key = parseString();
|
|
104
|
+
next(); // :
|
|
105
|
+
map.put(key, parseValue());
|
|
106
|
+
if (peek() == ',') next();
|
|
107
|
+
}
|
|
108
|
+
next(); // }
|
|
109
|
+
return map;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
List<Object> parseArray() {
|
|
113
|
+
List<Object> list = new ArrayList<>();
|
|
114
|
+
next(); // [
|
|
115
|
+
while (peek() != ']') {
|
|
116
|
+
list.add(parseValue());
|
|
117
|
+
if (peek() == ',') next();
|
|
118
|
+
}
|
|
119
|
+
next(); // ]
|
|
120
|
+
return list;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
String parseString() {
|
|
124
|
+
next(); // opening "
|
|
125
|
+
StringBuilder sb = new StringBuilder();
|
|
126
|
+
while (pos < s.length()) {
|
|
127
|
+
char c = s.charAt(pos++);
|
|
128
|
+
if (c == '"') break;
|
|
129
|
+
if (c == '\\' && pos < s.length()) {
|
|
130
|
+
char esc = s.charAt(pos++);
|
|
131
|
+
switch (esc) {
|
|
132
|
+
case '"': case '\\': case '/': sb.append(esc); break;
|
|
133
|
+
case 'n': sb.append('\n'); break;
|
|
134
|
+
case 't': sb.append('\t'); break;
|
|
135
|
+
case 'r': sb.append('\r'); break;
|
|
136
|
+
case 'u': sb.append((char) Integer.parseInt(s.substring(pos, pos+4), 16)); pos += 4; break;
|
|
137
|
+
default: sb.append(esc);
|
|
138
|
+
}
|
|
139
|
+
} else {
|
|
140
|
+
sb.append(c);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
return sb.toString();
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
Number parseNumber() {
|
|
147
|
+
int start = pos;
|
|
148
|
+
while (pos < s.length() && "0123456789.eE+-".indexOf(s.charAt(pos)) >= 0) pos++;
|
|
149
|
+
String num = s.substring(start, pos);
|
|
150
|
+
if (num.contains(".") || num.contains("e") || num.contains("E"))
|
|
151
|
+
return Double.parseDouble(num);
|
|
152
|
+
return Long.parseLong(num);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
// โโโ Helpers โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
|
|
157
|
+
|
|
158
|
+
static String esc(Object v) {
|
|
159
|
+
if (v == null) return "";
|
|
160
|
+
return v.toString().replace("&", "&").replace("<", "<").replace(">", ">").replace("\"", """);
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
static String badge(String status) {
|
|
164
|
+
Map<String, String> colors = Map.of("active","#22c55e","archived","#888","suspended","#ef4444","owner","#f59e0b","admin","#6366f1","member","#888");
|
|
165
|
+
String c = colors.getOrDefault(status, "#888");
|
|
166
|
+
return String.format("<span style='display:inline-block;padding:2px 10px;border-radius:999px;font-size:11px;font-weight:600;background:%s20;color:%s'>%s</span>", c, c, esc(status));
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
static int intVal(Map<String, Object> m, String key) {
|
|
170
|
+
Object v = m.get(key);
|
|
171
|
+
if (v instanceof Number) return ((Number) v).intValue();
|
|
172
|
+
return 0;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
static String strVal(Map<String, Object> m, String key) {
|
|
176
|
+
Object v = m.get(key);
|
|
177
|
+
return v != null ? v.toString() : "";
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
@SuppressWarnings("unchecked")
|
|
181
|
+
static List<Map<String, Object>> listVal(Map<String, Object> m, String key) {
|
|
182
|
+
Object v = m.get(key);
|
|
183
|
+
if (v instanceof List) return (List<Map<String, Object>>) v;
|
|
184
|
+
return List.of();
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
static String getSession(HttpExchange ex) {
|
|
188
|
+
String cookie = ex.getRequestHeaders().getFirst("Cookie");
|
|
189
|
+
if (cookie == null) return null;
|
|
190
|
+
for (String part : cookie.split(";")) {
|
|
191
|
+
part = part.trim();
|
|
192
|
+
if (part.startsWith("am_session=")) return part.substring(11);
|
|
193
|
+
}
|
|
194
|
+
return null;
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
static String getToken(HttpExchange ex) {
|
|
198
|
+
String sid = getSession(ex);
|
|
199
|
+
if (sid == null || !sessions.containsKey(sid)) return null;
|
|
200
|
+
return strVal(sessions.get(sid), "token");
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
static Map<String, Object> getUser(HttpExchange ex) {
|
|
204
|
+
String sid = getSession(ex);
|
|
205
|
+
if (sid == null || !sessions.containsKey(sid)) return null;
|
|
206
|
+
Object u = sessions.get(sid).get("user");
|
|
207
|
+
return u instanceof Map ? (Map<String, Object>) u : null;
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
static void respond(HttpExchange ex, int code, String html) throws IOException {
|
|
211
|
+
byte[] bytes = html.getBytes(StandardCharsets.UTF_8);
|
|
212
|
+
ex.getResponseHeaders().set("Content-Type", "text/html; charset=UTF-8");
|
|
213
|
+
ex.sendResponseHeaders(code, bytes.length);
|
|
214
|
+
ex.getResponseBody().write(bytes);
|
|
215
|
+
ex.getResponseBody().close();
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
static void redirect(HttpExchange ex, String url) throws IOException {
|
|
219
|
+
ex.getResponseHeaders().set("Location", url);
|
|
220
|
+
ex.sendResponseHeaders(302, -1);
|
|
221
|
+
ex.close();
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
static Map<String, String> parseForm(HttpExchange ex) throws IOException {
|
|
225
|
+
String body = new String(ex.getRequestBody().readAllBytes(), StandardCharsets.UTF_8);
|
|
226
|
+
Map<String, String> params = new LinkedHashMap<>();
|
|
227
|
+
for (String pair : body.split("&")) {
|
|
228
|
+
String[] kv = pair.split("=", 2);
|
|
229
|
+
if (kv.length == 2) params.put(URLDecoder.decode(kv[0], StandardCharsets.UTF_8), URLDecoder.decode(kv[1], StandardCharsets.UTF_8));
|
|
230
|
+
}
|
|
231
|
+
return params;
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
// โโโ CSS โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
|
|
235
|
+
|
|
236
|
+
static final String CSS = "*{box-sizing:border-box;margin:0;padding:0}:root{--bg:#0a0a0f;--surface:#12121a;--border:#1e1e2e;--text:#e4e4ef;--dim:#8888a0;--muted:#55556a;--primary:#6366f1;--success:#22c55e;--danger:#ef4444;--warning:#f59e0b}body{font-family:-apple-system,sans-serif;background:var(--bg);color:var(--text)}.layout{display:flex;min-height:100vh}.sidebar{width:240px;background:var(--surface);border-right:1px solid var(--border);position:fixed;top:0;left:0;bottom:0;display:flex;flex-direction:column}.sh{padding:20px;border-bottom:1px solid var(--border)}.sh h2{font-size:16px}.sh h2 em{font-style:normal;color:var(--primary)}.sh small{font-size:11px;color:var(--muted);display:block;margin-top:2px}.nav{flex:1;padding:8px 0}.ns{font-size:10px;text-transform:uppercase;letter-spacing:0.08em;color:var(--muted);padding:12px 20px 4px}.nav a{display:flex;align-items:center;gap:10px;padding:10px 20px;color:var(--dim);text-decoration:none;font-size:13px}.nav a:hover{color:var(--text);background:rgba(255,255,255,0.03)}.nav a.on{color:var(--primary);background:rgba(99,102,241,0.12);border-right:2px solid var(--primary)}.sf{padding:16px 20px;border-top:1px solid var(--border);font-size:12px}.content{flex:1;margin-left:240px;padding:32px;max-width:1100px}h2.t{font-size:22px;font-weight:700;margin-bottom:4px}.desc{font-size:13px;color:var(--dim);margin-bottom:24px}.stats{display:grid;grid-template-columns:repeat(auto-fit,minmax(180px,1fr));gap:16px;margin-bottom:24px}.stat{background:var(--surface);border:1px solid var(--border);border-radius:12px;padding:20px}.stat .l{font-size:11px;color:var(--muted);text-transform:uppercase;letter-spacing:0.06em}.stat .v{font-size:30px;font-weight:700;margin-top:4px}.card{background:var(--surface);border:1px solid var(--border);border-radius:12px;padding:20px;margin-bottom:16px}.ct{font-size:13px;color:var(--dim);text-transform:uppercase;letter-spacing:0.05em;font-weight:600;margin-bottom:12px}table{width:100%;border-collapse:collapse;font-size:13px}th{text-align:left;padding:10px 12px;color:var(--muted);font-size:11px;text-transform:uppercase;letter-spacing:0.05em;border-bottom:1px solid var(--border)}td{padding:12px;border-bottom:1px solid var(--border)}.btn{display:inline-flex;align-items:center;padding:8px 16px;border-radius:8px;font-size:13px;font-weight:600;cursor:pointer;border:1px solid var(--border);background:var(--surface);color:var(--text);text-decoration:none}.btn-p{background:var(--primary);border-color:var(--primary);color:#fff}.btn-d{color:var(--danger);border-color:var(--danger)}.btn-sm{padding:4px 10px;font-size:12px}.input{width:100%;padding:10px 14px;background:var(--bg);border:1px solid var(--border);border-radius:8px;color:var(--text);font-size:14px}.fg{margin-bottom:14px}.fl{display:block;font-size:12px;color:var(--dim);margin-bottom:4px}.empty{text-align:center;padding:48px 20px;color:var(--muted)}.empty-i{font-size:36px;margin-bottom:10px}select.input{appearance:auto}";
|
|
237
|
+
|
|
238
|
+
static String navItem(String href, String icon, String label, String active) {
|
|
239
|
+
String cls = active.equals(label.toLowerCase()) ? " on" : "";
|
|
240
|
+
return String.format("<a href='%s' class='%s'>%s <span>%s</span></a>", href, cls, icon, label);
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
static String layout(String page, Map<String, Object> user, String content) {
|
|
244
|
+
String userName = user != null ? esc(user.get("name")) : "";
|
|
245
|
+
String userEmail = user != null ? esc(user.get("email")) : "";
|
|
246
|
+
return String.format("<!DOCTYPE html><html><head><meta charset='UTF-8'><meta name='viewport' content='width=device-width,initial-scale=1.0'><title>AgenticMail Enterprise โ Java</title><style>%s</style></head><body><div class='layout'>" +
|
|
247
|
+
"<div class='sidebar'><div class='sh'><h2>๐ข <em>Agentic</em>Mail</h2><small>Enterprise ยท Java</small></div>" +
|
|
248
|
+
"<div class='nav'><div class='ns'>Overview</div>%s<div class='ns'>Manage</div>%s%s%s<div class='ns'>System</div>%s%s</div>" +
|
|
249
|
+
"<div class='sf'><div style='color:var(--dim)'>%s</div><div style='color:var(--muted);font-size:11px'>%s</div><a href='/logout' style='color:var(--muted);font-size:11px;margin-top:6px;display:inline-block'>Sign out</a></div></div>" +
|
|
250
|
+
"<div class='content'>%s</div></div></body></html>",
|
|
251
|
+
CSS,
|
|
252
|
+
navItem("/", "๐", "Dashboard", page), navItem("/agents", "๐ค", "Agents", page),
|
|
253
|
+
navItem("/users", "๐ฅ", "Users", page), navItem("/api-keys", "๐", "API Keys", page),
|
|
254
|
+
navItem("/audit", "๐", "Audit", page), navItem("/settings", "โ๏ธ", "Settings", page),
|
|
255
|
+
userName, userEmail, content);
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
// โโโ Handlers โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
|
|
259
|
+
|
|
260
|
+
static void handleLogin(HttpExchange ex) throws IOException {
|
|
261
|
+
if ("GET".equals(ex.getRequestMethod())) {
|
|
262
|
+
respond(ex, 200, String.format("<!DOCTYPE html><html><head><meta charset='UTF-8'><title>AgenticMail</title><style>%s</style></head><body style='display:flex;align-items:center;justify-content:center;min-height:100vh'><div style='width:380px'><h1 style='text-align:center;font-size:22px'>๐ข <em style='color:var(--primary)'>AgenticMail</em> Enterprise</h1><p style='text-align:center;color:var(--dim);font-size:13px;margin-bottom:32px'>Sign in ยท Java Dashboard</p><form method='POST'><div class='fg'><label class='fl'>Email</label><input class='input' type='email' name='email' required></div><div class='fg'><label class='fl'>Password</label><input class='input' type='password' name='password' required></div><button class='btn btn-p' style='width:100%%' type='submit'>Sign In</button></form></div></body></html>", CSS));
|
|
263
|
+
return;
|
|
264
|
+
}
|
|
265
|
+
Map<String, String> form = parseForm(ex);
|
|
266
|
+
var data = api("/auth/login", "POST", null, toJson(Map.of("email", form.getOrDefault("email", ""), "password", form.getOrDefault("password", ""))));
|
|
267
|
+
if (data.containsKey("token")) {
|
|
268
|
+
String sid = UUID.randomUUID().toString();
|
|
269
|
+
Map<String, Object> sess = new HashMap<>();
|
|
270
|
+
sess.put("token", data.get("token").toString());
|
|
271
|
+
sess.put("user", data.get("user"));
|
|
272
|
+
sessions.put(sid, sess);
|
|
273
|
+
ex.getResponseHeaders().set("Set-Cookie", "am_session=" + sid + "; Path=/; HttpOnly; Max-Age=86400");
|
|
274
|
+
redirect(ex, "/");
|
|
275
|
+
} else {
|
|
276
|
+
respond(ex, 401, "Login failed: " + strVal(data, "error") + " <a href='/login'>Try again</a>");
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
static void handleDashboard(HttpExchange ex) throws IOException {
|
|
281
|
+
String token = getToken(ex);
|
|
282
|
+
var stats = api("/api/stats", "GET", token, null);
|
|
283
|
+
var audit = api("/api/audit?limit=8", "GET", token, null);
|
|
284
|
+
|
|
285
|
+
StringBuilder events = new StringBuilder();
|
|
286
|
+
for (var e : listVal(audit, "events")) {
|
|
287
|
+
events.append(String.format("<div style='padding:10px 0;border-bottom:1px solid var(--border);font-size:13px'><span style='color:var(--primary);font-weight:500'>%s</span> on %s<div style='font-size:11px;color:var(--muted)'>%s</div></div>",
|
|
288
|
+
esc(e.get("action")), esc(e.get("resource")), esc(e.get("timestamp"))));
|
|
289
|
+
}
|
|
290
|
+
String evHtml = events.length() > 0 ? events.toString() : "<div class='empty'><div class='empty-i'>๐</div>No activity yet</div>";
|
|
291
|
+
|
|
292
|
+
respond(ex, 200, layout("dashboard", getUser(ex), String.format(
|
|
293
|
+
"<h2 class='t'>Dashboard</h2><p class='desc'>Overview</p><div class='stats'>" +
|
|
294
|
+
"<div class='stat'><div class='l'>Total Agents</div><div class='v' style='color:var(--primary)'>%d</div></div>" +
|
|
295
|
+
"<div class='stat'><div class='l'>Active Agents</div><div class='v' style='color:var(--success)'>%d</div></div>" +
|
|
296
|
+
"<div class='stat'><div class='l'>Users</div><div class='v'>%d</div></div>" +
|
|
297
|
+
"<div class='stat'><div class='l'>Audit Events</div><div class='v'>%d</div></div></div>" +
|
|
298
|
+
"<div class='card'><div class='ct'>Recent Activity</div>%s</div>",
|
|
299
|
+
intVal(stats, "totalAgents"), intVal(stats, "activeAgents"), intVal(stats, "totalUsers"), intVal(stats, "totalAuditEvents"), evHtml)));
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
static void handleAgents(HttpExchange ex) throws IOException {
|
|
303
|
+
String token = getToken(ex);
|
|
304
|
+
if ("POST".equals(ex.getRequestMethod())) {
|
|
305
|
+
Map<String, String> form = parseForm(ex);
|
|
306
|
+
api("/api/agents", "POST", token, toJson(Map.of("name", form.getOrDefault("name",""), "role", form.getOrDefault("role","assistant"))));
|
|
307
|
+
redirect(ex, "/agents"); return;
|
|
308
|
+
}
|
|
309
|
+
var data = api("/api/agents", "GET", token, null);
|
|
310
|
+
var agents = listVal(data, "agents");
|
|
311
|
+
StringBuilder rows = new StringBuilder();
|
|
312
|
+
for (var a : agents) {
|
|
313
|
+
rows.append(String.format("<tr><td style='font-weight:600'>%s</td><td style='color:var(--dim)'>%s</td><td>%s</td><td>%s</td></tr>",
|
|
314
|
+
esc(a.get("name")), esc(a.get("email")), esc(a.get("role")), badge(strVal(a, "status"))));
|
|
315
|
+
}
|
|
316
|
+
String table = agents.isEmpty() ? "<div class='empty'><div class='empty-i'>๐ค</div>No agents yet</div>" :
|
|
317
|
+
"<table><thead><tr><th>Name</th><th>Email</th><th>Role</th><th>Status</th></tr></thead><tbody>" + rows + "</tbody></table>";
|
|
318
|
+
respond(ex, 200, layout("agents", getUser(ex),
|
|
319
|
+
"<h2 class='t'>Agents</h2><p class='desc'>Manage AI agent identities</p>" +
|
|
320
|
+
"<div class='card' style='margin-bottom:16px'><div class='ct'>Create Agent</div><form method='POST' style='display:flex;gap:10px;align-items:end'>" +
|
|
321
|
+
"<div class='fg' style='flex:1;margin:0'><label class='fl'>Name</label><input class='input' name='name' required placeholder='e.g. researcher'></div>" +
|
|
322
|
+
"<div class='fg' style='margin:0'><label class='fl'>Role</label><select class='input' name='role'><option>assistant</option><option>researcher</option><option>writer</option></select></div>" +
|
|
323
|
+
"<button class='btn btn-p' type='submit'>Create</button></form></div><div class='card'>" + table + "</div>"));
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
static void handleUsers(HttpExchange ex) throws IOException {
|
|
327
|
+
String token = getToken(ex);
|
|
328
|
+
var data = api("/api/users", "GET", token, null);
|
|
329
|
+
var users = listVal(data, "users");
|
|
330
|
+
StringBuilder rows = new StringBuilder();
|
|
331
|
+
for (var u : users) {
|
|
332
|
+
rows.append(String.format("<tr><td style='font-weight:600'>%s</td><td style='color:var(--dim)'>%s</td><td>%s</td></tr>",
|
|
333
|
+
esc(u.get("name")), esc(u.get("email")), badge(strVal(u, "role"))));
|
|
334
|
+
}
|
|
335
|
+
String table = users.isEmpty() ? "<div class='empty'><div class='empty-i'>๐ฅ</div>No users yet</div>" :
|
|
336
|
+
"<table><thead><tr><th>Name</th><th>Email</th><th>Role</th></tr></thead><tbody>" + rows + "</tbody></table>";
|
|
337
|
+
respond(ex, 200, layout("users", getUser(ex),
|
|
338
|
+
"<h2 class='t'>Users</h2><p class='desc'>Manage team members</p><div class='card'>" + table + "</div>"));
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
static void handleSettings(HttpExchange ex) throws IOException {
|
|
342
|
+
String token = getToken(ex);
|
|
343
|
+
var s = api("/api/settings", "GET", token, null);
|
|
344
|
+
respond(ex, 200, layout("settings", getUser(ex),
|
|
345
|
+
"<h2 class='t'>Settings</h2><p class='desc'>Configure your organization</p>" +
|
|
346
|
+
"<div class='card'><div class='ct'>General</div><div style='font-size:13px'>Name: " + esc(s.get("name")) +
|
|
347
|
+
"<br>Domain: " + esc(s.get("domain")) + "<br>Plan: " + badge(strVal(s, "plan").toUpperCase()) +
|
|
348
|
+
"<br>Subdomain: " + esc(s.get("subdomain")) + ".agenticmail.cloud</div></div>"));
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
// โโโ Main โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
|
|
352
|
+
|
|
353
|
+
public static void main(String[] args) throws Exception {
|
|
354
|
+
String envUrl = System.getenv("AGENTICMAIL_URL");
|
|
355
|
+
if (envUrl != null && !envUrl.isEmpty()) API_URL = envUrl;
|
|
356
|
+
|
|
357
|
+
int port = 8081;
|
|
358
|
+
String envPort = System.getenv("PORT");
|
|
359
|
+
if (envPort != null) port = Integer.parseInt(envPort);
|
|
360
|
+
|
|
361
|
+
var server = HttpServer.create(new InetSocketAddress(port), 0);
|
|
362
|
+
server.setExecutor(Executors.newFixedThreadPool(10));
|
|
363
|
+
|
|
364
|
+
server.createContext("/login", ex -> { try { handleLogin(ex); } catch (Exception e) { respond(ex, 500, e.getMessage()); } });
|
|
365
|
+
server.createContext("/logout", ex -> { sessions.remove(getSession(ex)); ex.getResponseHeaders().set("Set-Cookie", "am_session=; Path=/; Max-Age=0"); redirect(ex, "/login"); });
|
|
366
|
+
server.createContext("/agents", ex -> { if (getToken(ex)==null) { redirect(ex,"/login"); return; } try { handleAgents(ex); } catch (Exception e) { respond(ex, 500, e.getMessage()); } });
|
|
367
|
+
server.createContext("/users", ex -> { if (getToken(ex)==null) { redirect(ex,"/login"); return; } try { handleUsers(ex); } catch (Exception e) { respond(ex, 500, e.getMessage()); } });
|
|
368
|
+
server.createContext("/settings", ex -> { if (getToken(ex)==null) { redirect(ex,"/login"); return; } try { handleSettings(ex); } catch (Exception e) { respond(ex, 500, e.getMessage()); } });
|
|
369
|
+
server.createContext("/", ex -> { if (getToken(ex)==null) { redirect(ex,"/login"); return; } try { handleDashboard(ex); } catch (Exception e) { respond(ex, 500, e.getMessage()); } });
|
|
370
|
+
|
|
371
|
+
server.start();
|
|
372
|
+
System.out.printf("%n๐ข AgenticMail Enterprise Dashboard (Java)%n");
|
|
373
|
+
System.out.printf(" API: %s%n", API_URL);
|
|
374
|
+
System.out.printf(" Dashboard: http://localhost:%d%n%n", port);
|
|
375
|
+
}
|
|
376
|
+
}
|