@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,319 @@
|
|
|
1
|
+
import {
|
|
2
|
+
DatabaseAdapter
|
|
3
|
+
} from "./chunk-77IDQJL3.js";
|
|
4
|
+
import "./chunk-PNKVD2UK.js";
|
|
5
|
+
|
|
6
|
+
// src/db/mongodb.ts
|
|
7
|
+
import { randomUUID, createHash } from "crypto";
|
|
8
|
+
var mongoMod;
|
|
9
|
+
async function getMongo() {
|
|
10
|
+
if (!mongoMod) {
|
|
11
|
+
try {
|
|
12
|
+
mongoMod = await import("mongodb");
|
|
13
|
+
} catch {
|
|
14
|
+
throw new Error("MongoDB driver not found. Install: npm install mongodb");
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
return mongoMod;
|
|
18
|
+
}
|
|
19
|
+
var MongoAdapter = class extends DatabaseAdapter {
|
|
20
|
+
type = "mongodb";
|
|
21
|
+
client = null;
|
|
22
|
+
db = null;
|
|
23
|
+
async connect(config) {
|
|
24
|
+
const { MongoClient } = await getMongo();
|
|
25
|
+
const uri = config.connectionString || `mongodb://${config.host || "localhost"}:${config.port || 27017}`;
|
|
26
|
+
this.client = new MongoClient(uri);
|
|
27
|
+
await this.client.connect();
|
|
28
|
+
const dbName = config.database || new URL(uri.replace("mongodb+srv://", "https://")).pathname.slice(1) || "agenticmail";
|
|
29
|
+
this.db = this.client.db(dbName);
|
|
30
|
+
}
|
|
31
|
+
async disconnect() {
|
|
32
|
+
if (this.client) await this.client.close();
|
|
33
|
+
}
|
|
34
|
+
isConnected() {
|
|
35
|
+
return this.client !== null;
|
|
36
|
+
}
|
|
37
|
+
col(name) {
|
|
38
|
+
return this.db.collection(name);
|
|
39
|
+
}
|
|
40
|
+
async migrate() {
|
|
41
|
+
await this.col("agents").createIndex({ name: 1 }, { unique: true });
|
|
42
|
+
await this.col("agents").createIndex({ email: 1 }, { unique: true });
|
|
43
|
+
await this.col("agents").createIndex({ status: 1 });
|
|
44
|
+
await this.col("users").createIndex({ email: 1 }, { unique: true });
|
|
45
|
+
await this.col("users").createIndex({ ssoProvider: 1, ssoSubject: 1 });
|
|
46
|
+
await this.col("audit_log").createIndex({ timestamp: -1 });
|
|
47
|
+
await this.col("audit_log").createIndex({ actor: 1 });
|
|
48
|
+
await this.col("audit_log").createIndex({ action: 1 });
|
|
49
|
+
await this.col("api_keys").createIndex({ keyHash: 1 }, { unique: true });
|
|
50
|
+
await this.col("email_rules").createIndex({ agentId: 1 });
|
|
51
|
+
await this.col("settings").updateOne(
|
|
52
|
+
{ _id: "default" },
|
|
53
|
+
{ $setOnInsert: { name: "", subdomain: "", plan: "free", primaryColor: "#6366f1", createdAt: /* @__PURE__ */ new Date(), updatedAt: /* @__PURE__ */ new Date() } },
|
|
54
|
+
{ upsert: true }
|
|
55
|
+
);
|
|
56
|
+
await this.col("retention_policy").updateOne(
|
|
57
|
+
{ _id: "default" },
|
|
58
|
+
{ $setOnInsert: { enabled: false, retainDays: 365, excludeTags: [], archiveFirst: true } },
|
|
59
|
+
{ upsert: true }
|
|
60
|
+
);
|
|
61
|
+
}
|
|
62
|
+
// ─── Company ─────────────────────────────────────────────
|
|
63
|
+
async getSettings() {
|
|
64
|
+
const r = await this.col("settings").findOne({ _id: "default" });
|
|
65
|
+
if (!r) return null;
|
|
66
|
+
return { id: "default", name: r.name, domain: r.domain, subdomain: r.subdomain, smtpHost: r.smtpHost, smtpPort: r.smtpPort, smtpUser: r.smtpUser, smtpPass: r.smtpPass, dkimPrivateKey: r.dkimPrivateKey, logoUrl: r.logoUrl, primaryColor: r.primaryColor, plan: r.plan, createdAt: r.createdAt, updatedAt: r.updatedAt };
|
|
67
|
+
}
|
|
68
|
+
async updateSettings(updates) {
|
|
69
|
+
const { id, ...rest } = updates;
|
|
70
|
+
await this.col("settings").updateOne({ _id: "default" }, { $set: { ...rest, updatedAt: /* @__PURE__ */ new Date() } }, { upsert: true });
|
|
71
|
+
return this.getSettings();
|
|
72
|
+
}
|
|
73
|
+
// ─── Agents ──────────────────────────────────────────────
|
|
74
|
+
async createAgent(input) {
|
|
75
|
+
const doc = {
|
|
76
|
+
_id: randomUUID(),
|
|
77
|
+
name: input.name,
|
|
78
|
+
email: input.email || `${input.name.toLowerCase().replace(/\s+/g, "-")}@localhost`,
|
|
79
|
+
role: input.role || "assistant",
|
|
80
|
+
status: "active",
|
|
81
|
+
metadata: input.metadata || {},
|
|
82
|
+
createdBy: input.createdBy,
|
|
83
|
+
createdAt: /* @__PURE__ */ new Date(),
|
|
84
|
+
updatedAt: /* @__PURE__ */ new Date()
|
|
85
|
+
};
|
|
86
|
+
await this.col("agents").insertOne(doc);
|
|
87
|
+
return this.docToAgent(doc);
|
|
88
|
+
}
|
|
89
|
+
async getAgent(id) {
|
|
90
|
+
const r = await this.col("agents").findOne({ _id: id });
|
|
91
|
+
return r ? this.docToAgent(r) : null;
|
|
92
|
+
}
|
|
93
|
+
async getAgentByName(name) {
|
|
94
|
+
const r = await this.col("agents").findOne({ name });
|
|
95
|
+
return r ? this.docToAgent(r) : null;
|
|
96
|
+
}
|
|
97
|
+
async listAgents(opts) {
|
|
98
|
+
const filter = {};
|
|
99
|
+
if (opts?.status) filter.status = opts.status;
|
|
100
|
+
const cursor = this.col("agents").find(filter).sort({ createdAt: -1 });
|
|
101
|
+
if (opts?.offset) cursor.skip(opts.offset);
|
|
102
|
+
if (opts?.limit) cursor.limit(opts.limit);
|
|
103
|
+
return (await cursor.toArray()).map((r) => this.docToAgent(r));
|
|
104
|
+
}
|
|
105
|
+
async updateAgent(id, updates) {
|
|
106
|
+
const set = { updatedAt: /* @__PURE__ */ new Date() };
|
|
107
|
+
for (const key of ["name", "email", "role", "status", "metadata"]) {
|
|
108
|
+
if (updates[key] !== void 0) set[key] = updates[key];
|
|
109
|
+
}
|
|
110
|
+
await this.col("agents").updateOne({ _id: id }, { $set: set });
|
|
111
|
+
return await this.getAgent(id);
|
|
112
|
+
}
|
|
113
|
+
async archiveAgent(id) {
|
|
114
|
+
await this.col("agents").updateOne({ _id: id }, { $set: { status: "archived", updatedAt: /* @__PURE__ */ new Date() } });
|
|
115
|
+
}
|
|
116
|
+
async deleteAgent(id) {
|
|
117
|
+
await this.col("agents").deleteOne({ _id: id });
|
|
118
|
+
}
|
|
119
|
+
async countAgents(status) {
|
|
120
|
+
const filter = {};
|
|
121
|
+
if (status) filter.status = status;
|
|
122
|
+
return this.col("agents").countDocuments(filter);
|
|
123
|
+
}
|
|
124
|
+
// ─── Users ───────────────────────────────────────────────
|
|
125
|
+
async createUser(input) {
|
|
126
|
+
let passwordHash = null;
|
|
127
|
+
if (input.password) {
|
|
128
|
+
const { default: bcrypt } = await import("bcryptjs");
|
|
129
|
+
passwordHash = await bcrypt.hash(input.password, 12);
|
|
130
|
+
}
|
|
131
|
+
const doc = {
|
|
132
|
+
_id: randomUUID(),
|
|
133
|
+
email: input.email,
|
|
134
|
+
name: input.name,
|
|
135
|
+
role: input.role,
|
|
136
|
+
passwordHash,
|
|
137
|
+
ssoProvider: input.ssoProvider || null,
|
|
138
|
+
ssoSubject: input.ssoSubject || null,
|
|
139
|
+
createdAt: /* @__PURE__ */ new Date(),
|
|
140
|
+
updatedAt: /* @__PURE__ */ new Date(),
|
|
141
|
+
lastLoginAt: null
|
|
142
|
+
};
|
|
143
|
+
await this.col("users").insertOne(doc);
|
|
144
|
+
return this.docToUser(doc);
|
|
145
|
+
}
|
|
146
|
+
async getUser(id) {
|
|
147
|
+
const r = await this.col("users").findOne({ _id: id });
|
|
148
|
+
return r ? this.docToUser(r) : null;
|
|
149
|
+
}
|
|
150
|
+
async getUserByEmail(email) {
|
|
151
|
+
const r = await this.col("users").findOne({ email });
|
|
152
|
+
return r ? this.docToUser(r) : null;
|
|
153
|
+
}
|
|
154
|
+
async getUserBySso(provider, subject) {
|
|
155
|
+
const r = await this.col("users").findOne({ ssoProvider: provider, ssoSubject: subject });
|
|
156
|
+
return r ? this.docToUser(r) : null;
|
|
157
|
+
}
|
|
158
|
+
async listUsers(opts) {
|
|
159
|
+
const cursor = this.col("users").find({}).sort({ createdAt: -1 });
|
|
160
|
+
if (opts?.offset) cursor.skip(opts.offset);
|
|
161
|
+
if (opts?.limit) cursor.limit(opts.limit);
|
|
162
|
+
return (await cursor.toArray()).map((r) => this.docToUser(r));
|
|
163
|
+
}
|
|
164
|
+
async updateUser(id, updates) {
|
|
165
|
+
const set = { updatedAt: /* @__PURE__ */ new Date() };
|
|
166
|
+
for (const key of ["email", "name", "role", "lastLoginAt"]) {
|
|
167
|
+
if (updates[key] !== void 0) set[key] = updates[key];
|
|
168
|
+
}
|
|
169
|
+
await this.col("users").updateOne({ _id: id }, { $set: set });
|
|
170
|
+
return await this.getUser(id);
|
|
171
|
+
}
|
|
172
|
+
async deleteUser(id) {
|
|
173
|
+
await this.col("users").deleteOne({ _id: id });
|
|
174
|
+
}
|
|
175
|
+
// ─── Audit ───────────────────────────────────────────────
|
|
176
|
+
async logEvent(event) {
|
|
177
|
+
await this.col("audit_log").insertOne({
|
|
178
|
+
_id: randomUUID(),
|
|
179
|
+
timestamp: /* @__PURE__ */ new Date(),
|
|
180
|
+
actor: event.actor,
|
|
181
|
+
actorType: event.actorType,
|
|
182
|
+
action: event.action,
|
|
183
|
+
resource: event.resource,
|
|
184
|
+
details: event.details || {},
|
|
185
|
+
ip: event.ip || null
|
|
186
|
+
});
|
|
187
|
+
}
|
|
188
|
+
async queryAudit(filters) {
|
|
189
|
+
const filter = {};
|
|
190
|
+
if (filters.actor) filter.actor = filters.actor;
|
|
191
|
+
if (filters.action) filter.action = filters.action;
|
|
192
|
+
if (filters.resource) filter.resource = { $regex: filters.resource, $options: "i" };
|
|
193
|
+
if (filters.from || filters.to) {
|
|
194
|
+
filter.timestamp = {};
|
|
195
|
+
if (filters.from) filter.timestamp.$gte = filters.from;
|
|
196
|
+
if (filters.to) filter.timestamp.$lte = filters.to;
|
|
197
|
+
}
|
|
198
|
+
const total = await this.col("audit_log").countDocuments(filter);
|
|
199
|
+
const cursor = this.col("audit_log").find(filter).sort({ timestamp: -1 });
|
|
200
|
+
if (filters.offset) cursor.skip(filters.offset);
|
|
201
|
+
if (filters.limit) cursor.limit(filters.limit);
|
|
202
|
+
const rows = await cursor.toArray();
|
|
203
|
+
return {
|
|
204
|
+
events: rows.map((r) => ({
|
|
205
|
+
id: r._id,
|
|
206
|
+
timestamp: r.timestamp,
|
|
207
|
+
actor: r.actor,
|
|
208
|
+
actorType: r.actorType,
|
|
209
|
+
action: r.action,
|
|
210
|
+
resource: r.resource,
|
|
211
|
+
details: r.details,
|
|
212
|
+
ip: r.ip
|
|
213
|
+
})),
|
|
214
|
+
total
|
|
215
|
+
};
|
|
216
|
+
}
|
|
217
|
+
// ─── API Keys ────────────────────────────────────────────
|
|
218
|
+
async createApiKey(input) {
|
|
219
|
+
const id = randomUUID();
|
|
220
|
+
const plaintext = `ek_${randomUUID().replace(/-/g, "")}`;
|
|
221
|
+
const keyHash = createHash("sha256").update(plaintext).digest("hex");
|
|
222
|
+
const keyPrefix = plaintext.substring(0, 11);
|
|
223
|
+
const doc = {
|
|
224
|
+
_id: id,
|
|
225
|
+
name: input.name,
|
|
226
|
+
keyHash,
|
|
227
|
+
keyPrefix,
|
|
228
|
+
scopes: input.scopes,
|
|
229
|
+
createdBy: input.createdBy,
|
|
230
|
+
createdAt: /* @__PURE__ */ new Date(),
|
|
231
|
+
lastUsedAt: null,
|
|
232
|
+
expiresAt: input.expiresAt || null,
|
|
233
|
+
revoked: false
|
|
234
|
+
};
|
|
235
|
+
await this.col("api_keys").insertOne(doc);
|
|
236
|
+
return { key: this.docToApiKey(doc), plaintext };
|
|
237
|
+
}
|
|
238
|
+
async getApiKey(id) {
|
|
239
|
+
const r = await this.col("api_keys").findOne({ _id: id });
|
|
240
|
+
return r ? this.docToApiKey(r) : null;
|
|
241
|
+
}
|
|
242
|
+
async validateApiKey(plaintext) {
|
|
243
|
+
const keyHash = createHash("sha256").update(plaintext).digest("hex");
|
|
244
|
+
const r = await this.col("api_keys").findOne({ keyHash, revoked: false });
|
|
245
|
+
if (!r) return null;
|
|
246
|
+
const key = this.docToApiKey(r);
|
|
247
|
+
if (key.expiresAt && /* @__PURE__ */ new Date() > key.expiresAt) return null;
|
|
248
|
+
await this.col("api_keys").updateOne({ _id: r._id }, { $set: { lastUsedAt: /* @__PURE__ */ new Date() } });
|
|
249
|
+
return key;
|
|
250
|
+
}
|
|
251
|
+
async listApiKeys(opts) {
|
|
252
|
+
const filter = {};
|
|
253
|
+
if (opts?.createdBy) filter.createdBy = opts.createdBy;
|
|
254
|
+
return (await this.col("api_keys").find(filter).sort({ createdAt: -1 }).toArray()).map((r) => this.docToApiKey(r));
|
|
255
|
+
}
|
|
256
|
+
async revokeApiKey(id) {
|
|
257
|
+
await this.col("api_keys").updateOne({ _id: id }, { $set: { revoked: true } });
|
|
258
|
+
}
|
|
259
|
+
// ─── Rules ───────────────────────────────────────────────
|
|
260
|
+
async createRule(rule) {
|
|
261
|
+
const doc = {
|
|
262
|
+
_id: randomUUID(),
|
|
263
|
+
...rule,
|
|
264
|
+
createdAt: /* @__PURE__ */ new Date(),
|
|
265
|
+
updatedAt: /* @__PURE__ */ new Date()
|
|
266
|
+
};
|
|
267
|
+
await this.col("email_rules").insertOne(doc);
|
|
268
|
+
return this.docToRule(doc);
|
|
269
|
+
}
|
|
270
|
+
async getRules(agentId) {
|
|
271
|
+
const filter = {};
|
|
272
|
+
if (agentId) filter.$or = [{ agentId }, { agentId: null }];
|
|
273
|
+
return (await this.col("email_rules").find(filter).sort({ priority: -1 }).toArray()).map((r) => this.docToRule(r));
|
|
274
|
+
}
|
|
275
|
+
async updateRule(id, updates) {
|
|
276
|
+
const { id: _id, createdAt, ...rest } = updates;
|
|
277
|
+
await this.col("email_rules").updateOne({ _id: id }, { $set: { ...rest, updatedAt: /* @__PURE__ */ new Date() } });
|
|
278
|
+
const r = await this.col("email_rules").findOne({ _id: id });
|
|
279
|
+
return this.docToRule(r);
|
|
280
|
+
}
|
|
281
|
+
async deleteRule(id) {
|
|
282
|
+
await this.col("email_rules").deleteOne({ _id: id });
|
|
283
|
+
}
|
|
284
|
+
// ─── Retention ───────────────────────────────────────────
|
|
285
|
+
async getRetentionPolicy() {
|
|
286
|
+
const r = await this.col("retention_policy").findOne({ _id: "default" });
|
|
287
|
+
if (!r) return { enabled: false, retainDays: 365, archiveFirst: true };
|
|
288
|
+
return { enabled: r.enabled, retainDays: r.retainDays, excludeTags: r.excludeTags || [], archiveFirst: r.archiveFirst };
|
|
289
|
+
}
|
|
290
|
+
async setRetentionPolicy(policy) {
|
|
291
|
+
await this.col("retention_policy").updateOne({ _id: "default" }, { $set: policy }, { upsert: true });
|
|
292
|
+
}
|
|
293
|
+
// ─── Stats ───────────────────────────────────────────────
|
|
294
|
+
async getStats() {
|
|
295
|
+
const [totalAgents, activeAgents, totalUsers, totalAuditEvents] = await Promise.all([
|
|
296
|
+
this.col("agents").countDocuments(),
|
|
297
|
+
this.col("agents").countDocuments({ status: "active" }),
|
|
298
|
+
this.col("users").countDocuments(),
|
|
299
|
+
this.col("audit_log").countDocuments()
|
|
300
|
+
]);
|
|
301
|
+
return { totalAgents, activeAgents, totalUsers, totalEmails: 0, totalAuditEvents };
|
|
302
|
+
}
|
|
303
|
+
// ─── Mappers ─────────────────────────────────────────────
|
|
304
|
+
docToAgent(r) {
|
|
305
|
+
return { id: r._id, name: r.name, email: r.email, role: r.role, status: r.status, metadata: r.metadata || {}, createdBy: r.createdBy, createdAt: r.createdAt, updatedAt: r.updatedAt };
|
|
306
|
+
}
|
|
307
|
+
docToUser(r) {
|
|
308
|
+
return { id: r._id, email: r.email, name: r.name, role: r.role, passwordHash: r.passwordHash, ssoProvider: r.ssoProvider, ssoSubject: r.ssoSubject, createdAt: r.createdAt, updatedAt: r.updatedAt, lastLoginAt: r.lastLoginAt || void 0 };
|
|
309
|
+
}
|
|
310
|
+
docToApiKey(r) {
|
|
311
|
+
return { id: r._id, name: r.name, keyHash: r.keyHash, keyPrefix: r.keyPrefix, scopes: r.scopes || [], createdBy: r.createdBy, createdAt: r.createdAt, lastUsedAt: r.lastUsedAt || void 0, expiresAt: r.expiresAt || void 0, revoked: r.revoked };
|
|
312
|
+
}
|
|
313
|
+
docToRule(r) {
|
|
314
|
+
return { id: r._id, name: r.name, agentId: r.agentId, conditions: r.conditions || {}, actions: r.actions || {}, priority: r.priority, enabled: r.enabled, createdAt: r.createdAt, updatedAt: r.updatedAt };
|
|
315
|
+
}
|
|
316
|
+
};
|
|
317
|
+
export {
|
|
318
|
+
MongoAdapter
|
|
319
|
+
};
|