@openacp/cli 0.2.16 → 0.2.18
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/dist/chunk-3UGSZYSQ.js +278 -0
- package/dist/chunk-3UGSZYSQ.js.map +1 -0
- package/dist/{chunk-TKOYKVXH.js → chunk-HTXK4NLG.js} +654 -82
- package/dist/chunk-HTXK4NLG.js.map +1 -0
- package/dist/chunk-ZATQZUJT.js +193 -0
- package/dist/chunk-ZATQZUJT.js.map +1 -0
- package/dist/cli.js +5 -2
- package/dist/cli.js.map +1 -1
- package/dist/index.d.ts +206 -15
- package/dist/index.js +10 -8
- package/dist/{main-C4QWS4PK.js → main-4H43GCXO.js} +16 -5
- package/dist/main-4H43GCXO.js.map +1 -0
- package/dist/{setup-5RQ7YLYW.js → setup-FB4DGR6R.js} +12 -2
- package/dist/{setup-5RQ7YLYW.js.map → setup-FB4DGR6R.js.map} +1 -1
- package/dist/tunnel-service-FPRPBPQ5.js +633 -0
- package/dist/tunnel-service-FPRPBPQ5.js.map +1 -0
- package/package.json +5 -1
- package/dist/chunk-HTUZOMIT.js +0 -421
- package/dist/chunk-HTUZOMIT.js.map +0 -1
- package/dist/chunk-TKOYKVXH.js.map +0 -1
- package/dist/main-C4QWS4PK.js.map +0 -1
|
@@ -0,0 +1,278 @@
|
|
|
1
|
+
import {
|
|
2
|
+
createChildLogger
|
|
3
|
+
} from "./chunk-ZATQZUJT.js";
|
|
4
|
+
|
|
5
|
+
// src/core/config.ts
|
|
6
|
+
import { z } from "zod";
|
|
7
|
+
import * as fs from "fs";
|
|
8
|
+
import * as path from "path";
|
|
9
|
+
import * as os from "os";
|
|
10
|
+
var log = createChildLogger({ module: "config" });
|
|
11
|
+
var BaseChannelSchema = z.object({
|
|
12
|
+
enabled: z.boolean().default(false),
|
|
13
|
+
adapter: z.string().optional()
|
|
14
|
+
// package name for plugin adapters
|
|
15
|
+
}).passthrough();
|
|
16
|
+
var PLUGINS_DIR = path.join(os.homedir(), ".openacp", "plugins");
|
|
17
|
+
var AgentSchema = z.object({
|
|
18
|
+
command: z.string(),
|
|
19
|
+
args: z.array(z.string()).default([]),
|
|
20
|
+
workingDirectory: z.string().optional(),
|
|
21
|
+
env: z.record(z.string(), z.string()).default({})
|
|
22
|
+
});
|
|
23
|
+
var LoggingSchema = z.object({
|
|
24
|
+
level: z.enum(["silent", "debug", "info", "warn", "error", "fatal"]).default("info"),
|
|
25
|
+
logDir: z.string().default("~/.openacp/logs"),
|
|
26
|
+
maxFileSize: z.union([z.string(), z.number()]).default("10m"),
|
|
27
|
+
maxFiles: z.number().default(7),
|
|
28
|
+
sessionLogRetentionDays: z.number().default(30)
|
|
29
|
+
}).default({});
|
|
30
|
+
var TunnelAuthSchema = z.object({
|
|
31
|
+
enabled: z.boolean().default(false),
|
|
32
|
+
token: z.string().optional()
|
|
33
|
+
}).default({});
|
|
34
|
+
var TunnelSchema = z.object({
|
|
35
|
+
enabled: z.boolean().default(false),
|
|
36
|
+
port: z.number().default(3100),
|
|
37
|
+
provider: z.enum(["cloudflare", "ngrok", "bore"]).default("cloudflare"),
|
|
38
|
+
options: z.record(z.string(), z.unknown()).default({}),
|
|
39
|
+
storeTtlMinutes: z.number().default(60),
|
|
40
|
+
auth: TunnelAuthSchema
|
|
41
|
+
}).default({});
|
|
42
|
+
var ConfigSchema = z.object({
|
|
43
|
+
channels: z.record(z.string(), BaseChannelSchema),
|
|
44
|
+
agents: z.record(z.string(), AgentSchema),
|
|
45
|
+
defaultAgent: z.string(),
|
|
46
|
+
workspace: z.object({
|
|
47
|
+
baseDir: z.string().default("~/openacp-workspace")
|
|
48
|
+
}).default({}),
|
|
49
|
+
security: z.object({
|
|
50
|
+
allowedUserIds: z.array(z.string()).default([]),
|
|
51
|
+
maxConcurrentSessions: z.number().default(5),
|
|
52
|
+
sessionTimeoutMinutes: z.number().default(60)
|
|
53
|
+
}).default({}),
|
|
54
|
+
logging: LoggingSchema,
|
|
55
|
+
sessionStore: z.object({
|
|
56
|
+
ttlDays: z.number().default(30)
|
|
57
|
+
}).default({}),
|
|
58
|
+
tunnel: TunnelSchema
|
|
59
|
+
});
|
|
60
|
+
function expandHome(p) {
|
|
61
|
+
if (p.startsWith("~")) {
|
|
62
|
+
return path.join(os.homedir(), p.slice(1));
|
|
63
|
+
}
|
|
64
|
+
return p;
|
|
65
|
+
}
|
|
66
|
+
var DEFAULT_CONFIG = {
|
|
67
|
+
channels: {
|
|
68
|
+
telegram: {
|
|
69
|
+
enabled: false,
|
|
70
|
+
botToken: "YOUR_BOT_TOKEN_HERE",
|
|
71
|
+
chatId: 0,
|
|
72
|
+
notificationTopicId: null,
|
|
73
|
+
assistantTopicId: null
|
|
74
|
+
}
|
|
75
|
+
},
|
|
76
|
+
agents: {
|
|
77
|
+
claude: { command: "claude-agent-acp", args: [], env: {} },
|
|
78
|
+
codex: { command: "codex", args: ["--acp"], env: {} }
|
|
79
|
+
},
|
|
80
|
+
defaultAgent: "claude",
|
|
81
|
+
workspace: { baseDir: "~/openacp-workspace" },
|
|
82
|
+
security: {
|
|
83
|
+
allowedUserIds: [],
|
|
84
|
+
maxConcurrentSessions: 5,
|
|
85
|
+
sessionTimeoutMinutes: 60
|
|
86
|
+
},
|
|
87
|
+
sessionStore: { ttlDays: 30 }
|
|
88
|
+
};
|
|
89
|
+
var ConfigManager = class {
|
|
90
|
+
config;
|
|
91
|
+
configPath;
|
|
92
|
+
constructor() {
|
|
93
|
+
this.configPath = process.env.OPENACP_CONFIG_PATH || expandHome("~/.openacp/config.json");
|
|
94
|
+
}
|
|
95
|
+
async load() {
|
|
96
|
+
const dir = path.dirname(this.configPath);
|
|
97
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
98
|
+
if (!fs.existsSync(this.configPath)) {
|
|
99
|
+
fs.writeFileSync(
|
|
100
|
+
this.configPath,
|
|
101
|
+
JSON.stringify(DEFAULT_CONFIG, null, 2)
|
|
102
|
+
);
|
|
103
|
+
log.info({ configPath: this.configPath }, "Config created");
|
|
104
|
+
log.info(
|
|
105
|
+
"Please edit it with your Telegram bot token and chat ID, then restart."
|
|
106
|
+
);
|
|
107
|
+
process.exit(1);
|
|
108
|
+
}
|
|
109
|
+
const raw = JSON.parse(fs.readFileSync(this.configPath, "utf-8"));
|
|
110
|
+
this.applyEnvOverrides(raw);
|
|
111
|
+
const result = ConfigSchema.safeParse(raw);
|
|
112
|
+
if (!result.success) {
|
|
113
|
+
log.error("Config validation failed");
|
|
114
|
+
for (const issue of result.error.issues) {
|
|
115
|
+
log.error(
|
|
116
|
+
{ path: issue.path.join("."), message: issue.message },
|
|
117
|
+
"Validation error"
|
|
118
|
+
);
|
|
119
|
+
}
|
|
120
|
+
process.exit(1);
|
|
121
|
+
}
|
|
122
|
+
this.config = result.data;
|
|
123
|
+
}
|
|
124
|
+
get() {
|
|
125
|
+
return this.config;
|
|
126
|
+
}
|
|
127
|
+
async save(updates) {
|
|
128
|
+
const raw = JSON.parse(fs.readFileSync(this.configPath, "utf-8"));
|
|
129
|
+
this.deepMerge(raw, updates);
|
|
130
|
+
fs.writeFileSync(this.configPath, JSON.stringify(raw, null, 2));
|
|
131
|
+
const result = ConfigSchema.safeParse(raw);
|
|
132
|
+
if (result.success) {
|
|
133
|
+
this.config = result.data;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
resolveWorkspace(input) {
|
|
137
|
+
if (!input) {
|
|
138
|
+
const resolved2 = expandHome(this.config.workspace.baseDir);
|
|
139
|
+
fs.mkdirSync(resolved2, { recursive: true });
|
|
140
|
+
return resolved2;
|
|
141
|
+
}
|
|
142
|
+
if (input.startsWith("/") || input.startsWith("~")) {
|
|
143
|
+
const resolved2 = expandHome(input);
|
|
144
|
+
fs.mkdirSync(resolved2, { recursive: true });
|
|
145
|
+
return resolved2;
|
|
146
|
+
}
|
|
147
|
+
const name = input.toLowerCase();
|
|
148
|
+
const resolved = path.join(expandHome(this.config.workspace.baseDir), name);
|
|
149
|
+
fs.mkdirSync(resolved, { recursive: true });
|
|
150
|
+
return resolved;
|
|
151
|
+
}
|
|
152
|
+
async exists() {
|
|
153
|
+
return fs.existsSync(this.configPath);
|
|
154
|
+
}
|
|
155
|
+
getConfigPath() {
|
|
156
|
+
return this.configPath;
|
|
157
|
+
}
|
|
158
|
+
async writeNew(config) {
|
|
159
|
+
const dir = path.dirname(this.configPath);
|
|
160
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
161
|
+
fs.writeFileSync(this.configPath, JSON.stringify(config, null, 2));
|
|
162
|
+
}
|
|
163
|
+
applyEnvOverrides(raw) {
|
|
164
|
+
const overrides = [
|
|
165
|
+
["OPENACP_TELEGRAM_BOT_TOKEN", ["channels", "telegram", "botToken"]],
|
|
166
|
+
["OPENACP_TELEGRAM_CHAT_ID", ["channels", "telegram", "chatId"]],
|
|
167
|
+
["OPENACP_DEFAULT_AGENT", ["defaultAgent"]]
|
|
168
|
+
];
|
|
169
|
+
for (const [envVar, configPath] of overrides) {
|
|
170
|
+
const value = process.env[envVar];
|
|
171
|
+
if (value !== void 0) {
|
|
172
|
+
let target = raw;
|
|
173
|
+
for (let i = 0; i < configPath.length - 1; i++) {
|
|
174
|
+
if (!target[configPath[i]]) target[configPath[i]] = {};
|
|
175
|
+
target = target[configPath[i]];
|
|
176
|
+
}
|
|
177
|
+
const key = configPath[configPath.length - 1];
|
|
178
|
+
target[key] = key === "chatId" ? Number(value) : value;
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
if (process.env.OPENACP_LOG_LEVEL) {
|
|
182
|
+
raw.logging = raw.logging || {};
|
|
183
|
+
raw.logging.level = process.env.OPENACP_LOG_LEVEL;
|
|
184
|
+
}
|
|
185
|
+
if (process.env.OPENACP_LOG_DIR) {
|
|
186
|
+
raw.logging = raw.logging || {};
|
|
187
|
+
raw.logging.logDir = process.env.OPENACP_LOG_DIR;
|
|
188
|
+
}
|
|
189
|
+
if (process.env.OPENACP_DEBUG && !process.env.OPENACP_LOG_LEVEL) {
|
|
190
|
+
raw.logging = raw.logging || {};
|
|
191
|
+
raw.logging.level = "debug";
|
|
192
|
+
}
|
|
193
|
+
if (process.env.OPENACP_TUNNEL_ENABLED) {
|
|
194
|
+
raw.tunnel = raw.tunnel || {};
|
|
195
|
+
raw.tunnel.enabled = process.env.OPENACP_TUNNEL_ENABLED === "true";
|
|
196
|
+
}
|
|
197
|
+
if (process.env.OPENACP_TUNNEL_PORT) {
|
|
198
|
+
raw.tunnel = raw.tunnel || {};
|
|
199
|
+
raw.tunnel.port = Number(
|
|
200
|
+
process.env.OPENACP_TUNNEL_PORT
|
|
201
|
+
);
|
|
202
|
+
}
|
|
203
|
+
if (process.env.OPENACP_TUNNEL_PROVIDER) {
|
|
204
|
+
raw.tunnel = raw.tunnel || {};
|
|
205
|
+
raw.tunnel.provider = process.env.OPENACP_TUNNEL_PROVIDER;
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
deepMerge(target, source) {
|
|
209
|
+
for (const key of Object.keys(source)) {
|
|
210
|
+
if (source[key] && typeof source[key] === "object" && !Array.isArray(source[key])) {
|
|
211
|
+
if (!target[key]) target[key] = {};
|
|
212
|
+
this.deepMerge(target[key], source[key]);
|
|
213
|
+
} else {
|
|
214
|
+
target[key] = source[key];
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
};
|
|
219
|
+
|
|
220
|
+
// src/core/plugin-manager.ts
|
|
221
|
+
import { execSync } from "child_process";
|
|
222
|
+
import * as fs2 from "fs";
|
|
223
|
+
import * as path2 from "path";
|
|
224
|
+
import { createRequire } from "module";
|
|
225
|
+
var log2 = createChildLogger({ module: "plugin-manager" });
|
|
226
|
+
function ensurePluginsDir() {
|
|
227
|
+
fs2.mkdirSync(PLUGINS_DIR, { recursive: true });
|
|
228
|
+
const pkgPath = path2.join(PLUGINS_DIR, "package.json");
|
|
229
|
+
if (!fs2.existsSync(pkgPath)) {
|
|
230
|
+
fs2.writeFileSync(pkgPath, JSON.stringify({ name: "openacp-plugins", private: true, dependencies: {} }, null, 2));
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
function installPlugin(packageName) {
|
|
234
|
+
ensurePluginsDir();
|
|
235
|
+
log2.info({ packageName }, "Installing plugin");
|
|
236
|
+
execSync(`npm install ${packageName} --prefix "${PLUGINS_DIR}"`, { stdio: "inherit" });
|
|
237
|
+
log2.info({ packageName }, "Plugin installed successfully");
|
|
238
|
+
}
|
|
239
|
+
function uninstallPlugin(packageName) {
|
|
240
|
+
ensurePluginsDir();
|
|
241
|
+
log2.info({ packageName }, "Uninstalling plugin");
|
|
242
|
+
execSync(`npm uninstall ${packageName} --prefix "${PLUGINS_DIR}"`, { stdio: "inherit" });
|
|
243
|
+
log2.info({ packageName }, "Plugin uninstalled");
|
|
244
|
+
}
|
|
245
|
+
function listPlugins() {
|
|
246
|
+
const pkgPath = path2.join(PLUGINS_DIR, "package.json");
|
|
247
|
+
if (!fs2.existsSync(pkgPath)) return {};
|
|
248
|
+
const pkg = JSON.parse(fs2.readFileSync(pkgPath, "utf-8"));
|
|
249
|
+
return pkg.dependencies || {};
|
|
250
|
+
}
|
|
251
|
+
async function loadAdapterFactory(packageName) {
|
|
252
|
+
try {
|
|
253
|
+
const require2 = createRequire(path2.join(PLUGINS_DIR, "package.json"));
|
|
254
|
+
const resolved = require2.resolve(packageName);
|
|
255
|
+
const mod = await import(resolved);
|
|
256
|
+
const factory = mod.adapterFactory || mod.default;
|
|
257
|
+
if (!factory || typeof factory.createAdapter !== "function") {
|
|
258
|
+
log2.error({ packageName }, "Plugin does not export a valid AdapterFactory (needs .createAdapter())");
|
|
259
|
+
return null;
|
|
260
|
+
}
|
|
261
|
+
return factory;
|
|
262
|
+
} catch (err) {
|
|
263
|
+
log2.error({ packageName, err }, "Failed to load plugin");
|
|
264
|
+
log2.error({ packageName }, "Run: npx openacp install <packageName>");
|
|
265
|
+
return null;
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
export {
|
|
270
|
+
PLUGINS_DIR,
|
|
271
|
+
expandHome,
|
|
272
|
+
ConfigManager,
|
|
273
|
+
installPlugin,
|
|
274
|
+
uninstallPlugin,
|
|
275
|
+
listPlugins,
|
|
276
|
+
loadAdapterFactory
|
|
277
|
+
};
|
|
278
|
+
//# sourceMappingURL=chunk-3UGSZYSQ.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/core/config.ts","../../src/core/plugin-manager.ts"],"sourcesContent":["import { z } from \"zod\";\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport * as os from \"node:os\";\nimport { createChildLogger } from \"./log.js\";\nconst log = createChildLogger({ module: \"config\" });\n\nconst BaseChannelSchema = z\n .object({\n enabled: z.boolean().default(false),\n adapter: z.string().optional(), // package name for plugin adapters\n })\n .passthrough();\n\nexport const PLUGINS_DIR = path.join(os.homedir(), \".openacp\", \"plugins\");\n\nconst AgentSchema = z.object({\n command: z.string(),\n args: z.array(z.string()).default([]),\n workingDirectory: z.string().optional(),\n env: z.record(z.string(), z.string()).default({}),\n});\n\nconst LoggingSchema = z\n .object({\n level: z\n .enum([\"silent\", \"debug\", \"info\", \"warn\", \"error\", \"fatal\"])\n .default(\"info\"),\n logDir: z.string().default(\"~/.openacp/logs\"),\n maxFileSize: z.union([z.string(), z.number()]).default(\"10m\"),\n maxFiles: z.number().default(7),\n sessionLogRetentionDays: z.number().default(30),\n })\n .default({});\n\nexport type LoggingConfig = z.infer<typeof LoggingSchema>;\n\nconst TunnelAuthSchema = z\n .object({\n enabled: z.boolean().default(false),\n token: z.string().optional(),\n })\n .default({});\n\nconst TunnelSchema = z\n .object({\n enabled: z.boolean().default(false),\n port: z.number().default(3100),\n provider: z.enum([\"cloudflare\", \"ngrok\", \"bore\"]).default(\"cloudflare\"),\n options: z.record(z.string(), z.unknown()).default({}),\n storeTtlMinutes: z.number().default(60),\n auth: TunnelAuthSchema,\n })\n .default({});\n\nexport type TunnelConfig = z.infer<typeof TunnelSchema>;\n\nexport const ConfigSchema = z.object({\n channels: z.record(z.string(), BaseChannelSchema),\n agents: z.record(z.string(), AgentSchema),\n defaultAgent: z.string(),\n workspace: z\n .object({\n baseDir: z.string().default(\"~/openacp-workspace\"),\n })\n .default({}),\n security: z\n .object({\n allowedUserIds: z.array(z.string()).default([]),\n maxConcurrentSessions: z.number().default(5),\n sessionTimeoutMinutes: z.number().default(60),\n })\n .default({}),\n logging: LoggingSchema,\n sessionStore: z\n .object({\n ttlDays: z.number().default(30),\n })\n .default({}),\n tunnel: TunnelSchema,\n});\n\nexport type Config = z.infer<typeof ConfigSchema>;\n\nexport function expandHome(p: string): string {\n if (p.startsWith(\"~\")) {\n return path.join(os.homedir(), p.slice(1));\n }\n return p;\n}\n\nconst DEFAULT_CONFIG = {\n channels: {\n telegram: {\n enabled: false,\n botToken: \"YOUR_BOT_TOKEN_HERE\",\n chatId: 0,\n notificationTopicId: null,\n assistantTopicId: null,\n },\n },\n agents: {\n claude: { command: \"claude-agent-acp\", args: [], env: {} },\n codex: { command: \"codex\", args: [\"--acp\"], env: {} },\n },\n defaultAgent: \"claude\",\n workspace: { baseDir: \"~/openacp-workspace\" },\n security: {\n allowedUserIds: [],\n maxConcurrentSessions: 5,\n sessionTimeoutMinutes: 60,\n },\n sessionStore: { ttlDays: 30 },\n};\n\nexport class ConfigManager {\n private config!: Config;\n private configPath: string;\n\n constructor() {\n this.configPath =\n process.env.OPENACP_CONFIG_PATH || expandHome(\"~/.openacp/config.json\");\n }\n\n async load(): Promise<void> {\n // 1. Ensure directory exists\n const dir = path.dirname(this.configPath);\n fs.mkdirSync(dir, { recursive: true });\n\n // 2. If config file doesn't exist, create default\n if (!fs.existsSync(this.configPath)) {\n fs.writeFileSync(\n this.configPath,\n JSON.stringify(DEFAULT_CONFIG, null, 2),\n );\n log.info({ configPath: this.configPath }, \"Config created\");\n log.info(\n \"Please edit it with your Telegram bot token and chat ID, then restart.\",\n );\n process.exit(1);\n }\n\n // 3. Read and parse\n const raw = JSON.parse(fs.readFileSync(this.configPath, \"utf-8\"));\n\n // 4. Apply env var overrides\n this.applyEnvOverrides(raw);\n\n // 5. Validate with Zod\n const result = ConfigSchema.safeParse(raw);\n if (!result.success) {\n log.error(\"Config validation failed\");\n for (const issue of result.error.issues) {\n log.error(\n { path: issue.path.join(\".\"), message: issue.message },\n \"Validation error\",\n );\n }\n process.exit(1);\n }\n this.config = result.data;\n }\n\n get(): Config {\n return this.config;\n }\n\n async save(updates: Record<string, unknown>): Promise<void> {\n // Read current file, merge updates, write back\n const raw = JSON.parse(fs.readFileSync(this.configPath, \"utf-8\"));\n this.deepMerge(raw, updates);\n fs.writeFileSync(this.configPath, JSON.stringify(raw, null, 2));\n // Re-validate and update in-memory config\n const result = ConfigSchema.safeParse(raw);\n if (result.success) {\n this.config = result.data;\n }\n }\n\n resolveWorkspace(input?: string): string {\n if (!input) {\n const resolved = expandHome(this.config.workspace.baseDir);\n fs.mkdirSync(resolved, { recursive: true });\n return resolved;\n }\n if (input.startsWith(\"/\") || input.startsWith(\"~\")) {\n const resolved = expandHome(input);\n fs.mkdirSync(resolved, { recursive: true });\n return resolved;\n }\n // Named workspace → lowercase, under baseDir\n const name = input.toLowerCase();\n const resolved = path.join(expandHome(this.config.workspace.baseDir), name);\n fs.mkdirSync(resolved, { recursive: true });\n return resolved;\n }\n\n async exists(): Promise<boolean> {\n return fs.existsSync(this.configPath);\n }\n\n getConfigPath(): string {\n return this.configPath;\n }\n\n async writeNew(config: Config): Promise<void> {\n const dir = path.dirname(this.configPath);\n fs.mkdirSync(dir, { recursive: true });\n fs.writeFileSync(this.configPath, JSON.stringify(config, null, 2));\n }\n\n private applyEnvOverrides(raw: Record<string, unknown>): void {\n const overrides: [string, string[]][] = [\n [\"OPENACP_TELEGRAM_BOT_TOKEN\", [\"channels\", \"telegram\", \"botToken\"]],\n [\"OPENACP_TELEGRAM_CHAT_ID\", [\"channels\", \"telegram\", \"chatId\"]],\n [\"OPENACP_DEFAULT_AGENT\", [\"defaultAgent\"]],\n ];\n for (const [envVar, configPath] of overrides) {\n const value = process.env[envVar];\n if (value !== undefined) {\n let target = raw as Record<string, any>;\n for (let i = 0; i < configPath.length - 1; i++) {\n if (!target[configPath[i]]) target[configPath[i]] = {};\n target = target[configPath[i]];\n }\n const key = configPath[configPath.length - 1];\n // Convert chatId to number\n target[key] = key === \"chatId\" ? Number(value) : value;\n }\n }\n\n // Logging env var overrides\n if (process.env.OPENACP_LOG_LEVEL) {\n raw.logging = raw.logging || {};\n (raw.logging as Record<string, unknown>).level =\n process.env.OPENACP_LOG_LEVEL;\n }\n if (process.env.OPENACP_LOG_DIR) {\n raw.logging = raw.logging || {};\n (raw.logging as Record<string, unknown>).logDir =\n process.env.OPENACP_LOG_DIR;\n }\n if (process.env.OPENACP_DEBUG && !process.env.OPENACP_LOG_LEVEL) {\n raw.logging = raw.logging || {};\n (raw.logging as Record<string, unknown>).level = \"debug\";\n }\n\n // Tunnel env var overrides\n if (process.env.OPENACP_TUNNEL_ENABLED) {\n raw.tunnel = raw.tunnel || {};\n (raw.tunnel as Record<string, unknown>).enabled =\n process.env.OPENACP_TUNNEL_ENABLED === \"true\";\n }\n if (process.env.OPENACP_TUNNEL_PORT) {\n raw.tunnel = raw.tunnel || {};\n (raw.tunnel as Record<string, unknown>).port = Number(\n process.env.OPENACP_TUNNEL_PORT,\n );\n }\n if (process.env.OPENACP_TUNNEL_PROVIDER) {\n raw.tunnel = raw.tunnel || {};\n (raw.tunnel as Record<string, unknown>).provider =\n process.env.OPENACP_TUNNEL_PROVIDER;\n }\n }\n\n private deepMerge(\n target: Record<string, any>,\n source: Record<string, any>,\n ): void {\n for (const key of Object.keys(source)) {\n if (\n source[key] &&\n typeof source[key] === \"object\" &&\n !Array.isArray(source[key])\n ) {\n if (!target[key]) target[key] = {};\n this.deepMerge(target[key], source[key]);\n } else {\n target[key] = source[key];\n }\n }\n }\n}\n","import { execSync } from 'node:child_process'\nimport * as fs from 'node:fs'\nimport * as path from 'node:path'\nimport { createRequire } from 'node:module'\nimport { PLUGINS_DIR } from './config.js'\nimport { createChildLogger } from './log.js'\nconst log = createChildLogger({ module: 'plugin-manager' })\nimport type { ChannelAdapter } from './channel.js'\n\nexport interface AdapterFactory {\n name: string\n createAdapter(core: any, config: any): ChannelAdapter\n}\n\nfunction ensurePluginsDir(): void {\n fs.mkdirSync(PLUGINS_DIR, { recursive: true })\n const pkgPath = path.join(PLUGINS_DIR, 'package.json')\n if (!fs.existsSync(pkgPath)) {\n fs.writeFileSync(pkgPath, JSON.stringify({ name: 'openacp-plugins', private: true, dependencies: {} }, null, 2))\n }\n}\n\nexport function installPlugin(packageName: string): void {\n ensurePluginsDir()\n log.info({ packageName }, 'Installing plugin')\n execSync(`npm install ${packageName} --prefix \"${PLUGINS_DIR}\"`, { stdio: 'inherit' })\n log.info({ packageName }, 'Plugin installed successfully')\n}\n\nexport function uninstallPlugin(packageName: string): void {\n ensurePluginsDir()\n log.info({ packageName }, 'Uninstalling plugin')\n execSync(`npm uninstall ${packageName} --prefix \"${PLUGINS_DIR}\"`, { stdio: 'inherit' })\n log.info({ packageName }, 'Plugin uninstalled')\n}\n\nexport function listPlugins(): Record<string, string> {\n const pkgPath = path.join(PLUGINS_DIR, 'package.json')\n if (!fs.existsSync(pkgPath)) return {}\n const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'))\n return pkg.dependencies || {}\n}\n\nexport async function loadAdapterFactory(packageName: string): Promise<AdapterFactory | null> {\n try {\n const require = createRequire(path.join(PLUGINS_DIR, 'package.json'))\n const resolved = require.resolve(packageName)\n const mod = await import(resolved)\n\n // Plugin must export `adapterFactory` or default export conforming to AdapterFactory\n const factory: AdapterFactory | undefined = mod.adapterFactory || mod.default\n if (!factory || typeof factory.createAdapter !== 'function') {\n log.error({ packageName }, 'Plugin does not export a valid AdapterFactory (needs .createAdapter())')\n return null\n }\n return factory\n } catch (err) {\n log.error({ packageName, err }, 'Failed to load plugin')\n log.error({ packageName }, 'Run: npx openacp install <packageName>')\n return null\n }\n}\n"],"mappings":";;;;;AAAA,SAAS,SAAS;AAClB,YAAY,QAAQ;AACpB,YAAY,UAAU;AACtB,YAAY,QAAQ;AAEpB,IAAM,MAAM,kBAAkB,EAAE,QAAQ,SAAS,CAAC;AAElD,IAAM,oBAAoB,EACvB,OAAO;AAAA,EACN,SAAS,EAAE,QAAQ,EAAE,QAAQ,KAAK;AAAA,EAClC,SAAS,EAAE,OAAO,EAAE,SAAS;AAAA;AAC/B,CAAC,EACA,YAAY;AAER,IAAM,cAAmB,UAAQ,WAAQ,GAAG,YAAY,SAAS;AAExE,IAAM,cAAc,EAAE,OAAO;AAAA,EAC3B,SAAS,EAAE,OAAO;AAAA,EAClB,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,CAAC,CAAC;AAAA,EACpC,kBAAkB,EAAE,OAAO,EAAE,SAAS;AAAA,EACtC,KAAK,EAAE,OAAO,EAAE,OAAO,GAAG,EAAE,OAAO,CAAC,EAAE,QAAQ,CAAC,CAAC;AAClD,CAAC;AAED,IAAM,gBAAgB,EACnB,OAAO;AAAA,EACN,OAAO,EACJ,KAAK,CAAC,UAAU,SAAS,QAAQ,QAAQ,SAAS,OAAO,CAAC,EAC1D,QAAQ,MAAM;AAAA,EACjB,QAAQ,EAAE,OAAO,EAAE,QAAQ,iBAAiB;AAAA,EAC5C,aAAa,EAAE,MAAM,CAAC,EAAE,OAAO,GAAG,EAAE,OAAO,CAAC,CAAC,EAAE,QAAQ,KAAK;AAAA,EAC5D,UAAU,EAAE,OAAO,EAAE,QAAQ,CAAC;AAAA,EAC9B,yBAAyB,EAAE,OAAO,EAAE,QAAQ,EAAE;AAChD,CAAC,EACA,QAAQ,CAAC,CAAC;AAIb,IAAM,mBAAmB,EACtB,OAAO;AAAA,EACN,SAAS,EAAE,QAAQ,EAAE,QAAQ,KAAK;AAAA,EAClC,OAAO,EAAE,OAAO,EAAE,SAAS;AAC7B,CAAC,EACA,QAAQ,CAAC,CAAC;AAEb,IAAM,eAAe,EAClB,OAAO;AAAA,EACN,SAAS,EAAE,QAAQ,EAAE,QAAQ,KAAK;AAAA,EAClC,MAAM,EAAE,OAAO,EAAE,QAAQ,IAAI;AAAA,EAC7B,UAAU,EAAE,KAAK,CAAC,cAAc,SAAS,MAAM,CAAC,EAAE,QAAQ,YAAY;AAAA,EACtE,SAAS,EAAE,OAAO,EAAE,OAAO,GAAG,EAAE,QAAQ,CAAC,EAAE,QAAQ,CAAC,CAAC;AAAA,EACrD,iBAAiB,EAAE,OAAO,EAAE,QAAQ,EAAE;AAAA,EACtC,MAAM;AACR,CAAC,EACA,QAAQ,CAAC,CAAC;AAIN,IAAM,eAAe,EAAE,OAAO;AAAA,EACnC,UAAU,EAAE,OAAO,EAAE,OAAO,GAAG,iBAAiB;AAAA,EAChD,QAAQ,EAAE,OAAO,EAAE,OAAO,GAAG,WAAW;AAAA,EACxC,cAAc,EAAE,OAAO;AAAA,EACvB,WAAW,EACR,OAAO;AAAA,IACN,SAAS,EAAE,OAAO,EAAE,QAAQ,qBAAqB;AAAA,EACnD,CAAC,EACA,QAAQ,CAAC,CAAC;AAAA,EACb,UAAU,EACP,OAAO;AAAA,IACN,gBAAgB,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,CAAC,CAAC;AAAA,IAC9C,uBAAuB,EAAE,OAAO,EAAE,QAAQ,CAAC;AAAA,IAC3C,uBAAuB,EAAE,OAAO,EAAE,QAAQ,EAAE;AAAA,EAC9C,CAAC,EACA,QAAQ,CAAC,CAAC;AAAA,EACb,SAAS;AAAA,EACT,cAAc,EACX,OAAO;AAAA,IACN,SAAS,EAAE,OAAO,EAAE,QAAQ,EAAE;AAAA,EAChC,CAAC,EACA,QAAQ,CAAC,CAAC;AAAA,EACb,QAAQ;AACV,CAAC;AAIM,SAAS,WAAW,GAAmB;AAC5C,MAAI,EAAE,WAAW,GAAG,GAAG;AACrB,WAAY,UAAQ,WAAQ,GAAG,EAAE,MAAM,CAAC,CAAC;AAAA,EAC3C;AACA,SAAO;AACT;AAEA,IAAM,iBAAiB;AAAA,EACrB,UAAU;AAAA,IACR,UAAU;AAAA,MACR,SAAS;AAAA,MACT,UAAU;AAAA,MACV,QAAQ;AAAA,MACR,qBAAqB;AAAA,MACrB,kBAAkB;AAAA,IACpB;AAAA,EACF;AAAA,EACA,QAAQ;AAAA,IACN,QAAQ,EAAE,SAAS,oBAAoB,MAAM,CAAC,GAAG,KAAK,CAAC,EAAE;AAAA,IACzD,OAAO,EAAE,SAAS,SAAS,MAAM,CAAC,OAAO,GAAG,KAAK,CAAC,EAAE;AAAA,EACtD;AAAA,EACA,cAAc;AAAA,EACd,WAAW,EAAE,SAAS,sBAAsB;AAAA,EAC5C,UAAU;AAAA,IACR,gBAAgB,CAAC;AAAA,IACjB,uBAAuB;AAAA,IACvB,uBAAuB;AAAA,EACzB;AAAA,EACA,cAAc,EAAE,SAAS,GAAG;AAC9B;AAEO,IAAM,gBAAN,MAAoB;AAAA,EACjB;AAAA,EACA;AAAA,EAER,cAAc;AACZ,SAAK,aACH,QAAQ,IAAI,uBAAuB,WAAW,wBAAwB;AAAA,EAC1E;AAAA,EAEA,MAAM,OAAsB;AAE1B,UAAM,MAAW,aAAQ,KAAK,UAAU;AACxC,IAAG,aAAU,KAAK,EAAE,WAAW,KAAK,CAAC;AAGrC,QAAI,CAAI,cAAW,KAAK,UAAU,GAAG;AACnC,MAAG;AAAA,QACD,KAAK;AAAA,QACL,KAAK,UAAU,gBAAgB,MAAM,CAAC;AAAA,MACxC;AACA,UAAI,KAAK,EAAE,YAAY,KAAK,WAAW,GAAG,gBAAgB;AAC1D,UAAI;AAAA,QACF;AAAA,MACF;AACA,cAAQ,KAAK,CAAC;AAAA,IAChB;AAGA,UAAM,MAAM,KAAK,MAAS,gBAAa,KAAK,YAAY,OAAO,CAAC;AAGhE,SAAK,kBAAkB,GAAG;AAG1B,UAAM,SAAS,aAAa,UAAU,GAAG;AACzC,QAAI,CAAC,OAAO,SAAS;AACnB,UAAI,MAAM,0BAA0B;AACpC,iBAAW,SAAS,OAAO,MAAM,QAAQ;AACvC,YAAI;AAAA,UACF,EAAE,MAAM,MAAM,KAAK,KAAK,GAAG,GAAG,SAAS,MAAM,QAAQ;AAAA,UACrD;AAAA,QACF;AAAA,MACF;AACA,cAAQ,KAAK,CAAC;AAAA,IAChB;AACA,SAAK,SAAS,OAAO;AAAA,EACvB;AAAA,EAEA,MAAc;AACZ,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,MAAM,KAAK,SAAiD;AAE1D,UAAM,MAAM,KAAK,MAAS,gBAAa,KAAK,YAAY,OAAO,CAAC;AAChE,SAAK,UAAU,KAAK,OAAO;AAC3B,IAAG,iBAAc,KAAK,YAAY,KAAK,UAAU,KAAK,MAAM,CAAC,CAAC;AAE9D,UAAM,SAAS,aAAa,UAAU,GAAG;AACzC,QAAI,OAAO,SAAS;AAClB,WAAK,SAAS,OAAO;AAAA,IACvB;AAAA,EACF;AAAA,EAEA,iBAAiB,OAAwB;AACvC,QAAI,CAAC,OAAO;AACV,YAAMA,YAAW,WAAW,KAAK,OAAO,UAAU,OAAO;AACzD,MAAG,aAAUA,WAAU,EAAE,WAAW,KAAK,CAAC;AAC1C,aAAOA;AAAA,IACT;AACA,QAAI,MAAM,WAAW,GAAG,KAAK,MAAM,WAAW,GAAG,GAAG;AAClD,YAAMA,YAAW,WAAW,KAAK;AACjC,MAAG,aAAUA,WAAU,EAAE,WAAW,KAAK,CAAC;AAC1C,aAAOA;AAAA,IACT;AAEA,UAAM,OAAO,MAAM,YAAY;AAC/B,UAAM,WAAgB,UAAK,WAAW,KAAK,OAAO,UAAU,OAAO,GAAG,IAAI;AAC1E,IAAG,aAAU,UAAU,EAAE,WAAW,KAAK,CAAC;AAC1C,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,SAA2B;AAC/B,WAAU,cAAW,KAAK,UAAU;AAAA,EACtC;AAAA,EAEA,gBAAwB;AACtB,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,MAAM,SAAS,QAA+B;AAC5C,UAAM,MAAW,aAAQ,KAAK,UAAU;AACxC,IAAG,aAAU,KAAK,EAAE,WAAW,KAAK,CAAC;AACrC,IAAG,iBAAc,KAAK,YAAY,KAAK,UAAU,QAAQ,MAAM,CAAC,CAAC;AAAA,EACnE;AAAA,EAEQ,kBAAkB,KAAoC;AAC5D,UAAM,YAAkC;AAAA,MACtC,CAAC,8BAA8B,CAAC,YAAY,YAAY,UAAU,CAAC;AAAA,MACnE,CAAC,4BAA4B,CAAC,YAAY,YAAY,QAAQ,CAAC;AAAA,MAC/D,CAAC,yBAAyB,CAAC,cAAc,CAAC;AAAA,IAC5C;AACA,eAAW,CAAC,QAAQ,UAAU,KAAK,WAAW;AAC5C,YAAM,QAAQ,QAAQ,IAAI,MAAM;AAChC,UAAI,UAAU,QAAW;AACvB,YAAI,SAAS;AACb,iBAAS,IAAI,GAAG,IAAI,WAAW,SAAS,GAAG,KAAK;AAC9C,cAAI,CAAC,OAAO,WAAW,CAAC,CAAC,EAAG,QAAO,WAAW,CAAC,CAAC,IAAI,CAAC;AACrD,mBAAS,OAAO,WAAW,CAAC,CAAC;AAAA,QAC/B;AACA,cAAM,MAAM,WAAW,WAAW,SAAS,CAAC;AAE5C,eAAO,GAAG,IAAI,QAAQ,WAAW,OAAO,KAAK,IAAI;AAAA,MACnD;AAAA,IACF;AAGA,QAAI,QAAQ,IAAI,mBAAmB;AACjC,UAAI,UAAU,IAAI,WAAW,CAAC;AAC9B,MAAC,IAAI,QAAoC,QACvC,QAAQ,IAAI;AAAA,IAChB;AACA,QAAI,QAAQ,IAAI,iBAAiB;AAC/B,UAAI,UAAU,IAAI,WAAW,CAAC;AAC9B,MAAC,IAAI,QAAoC,SACvC,QAAQ,IAAI;AAAA,IAChB;AACA,QAAI,QAAQ,IAAI,iBAAiB,CAAC,QAAQ,IAAI,mBAAmB;AAC/D,UAAI,UAAU,IAAI,WAAW,CAAC;AAC9B,MAAC,IAAI,QAAoC,QAAQ;AAAA,IACnD;AAGA,QAAI,QAAQ,IAAI,wBAAwB;AACtC,UAAI,SAAS,IAAI,UAAU,CAAC;AAC5B,MAAC,IAAI,OAAmC,UACtC,QAAQ,IAAI,2BAA2B;AAAA,IAC3C;AACA,QAAI,QAAQ,IAAI,qBAAqB;AACnC,UAAI,SAAS,IAAI,UAAU,CAAC;AAC5B,MAAC,IAAI,OAAmC,OAAO;AAAA,QAC7C,QAAQ,IAAI;AAAA,MACd;AAAA,IACF;AACA,QAAI,QAAQ,IAAI,yBAAyB;AACvC,UAAI,SAAS,IAAI,UAAU,CAAC;AAC5B,MAAC,IAAI,OAAmC,WACtC,QAAQ,IAAI;AAAA,IAChB;AAAA,EACF;AAAA,EAEQ,UACN,QACA,QACM;AACN,eAAW,OAAO,OAAO,KAAK,MAAM,GAAG;AACrC,UACE,OAAO,GAAG,KACV,OAAO,OAAO,GAAG,MAAM,YACvB,CAAC,MAAM,QAAQ,OAAO,GAAG,CAAC,GAC1B;AACA,YAAI,CAAC,OAAO,GAAG,EAAG,QAAO,GAAG,IAAI,CAAC;AACjC,aAAK,UAAU,OAAO,GAAG,GAAG,OAAO,GAAG,CAAC;AAAA,MACzC,OAAO;AACL,eAAO,GAAG,IAAI,OAAO,GAAG;AAAA,MAC1B;AAAA,IACF;AAAA,EACF;AACF;;;AC3RA,SAAS,gBAAgB;AACzB,YAAYC,SAAQ;AACpB,YAAYC,WAAU;AACtB,SAAS,qBAAqB;AAG9B,IAAMC,OAAM,kBAAkB,EAAE,QAAQ,iBAAiB,CAAC;AAQ1D,SAAS,mBAAyB;AAChC,EAAG,cAAU,aAAa,EAAE,WAAW,KAAK,CAAC;AAC7C,QAAM,UAAe,WAAK,aAAa,cAAc;AACrD,MAAI,CAAI,eAAW,OAAO,GAAG;AAC3B,IAAG,kBAAc,SAAS,KAAK,UAAU,EAAE,MAAM,mBAAmB,SAAS,MAAM,cAAc,CAAC,EAAE,GAAG,MAAM,CAAC,CAAC;AAAA,EACjH;AACF;AAEO,SAAS,cAAc,aAA2B;AACvD,mBAAiB;AACjB,EAAAA,KAAI,KAAK,EAAE,YAAY,GAAG,mBAAmB;AAC7C,WAAS,eAAe,WAAW,cAAc,WAAW,KAAK,EAAE,OAAO,UAAU,CAAC;AACrF,EAAAA,KAAI,KAAK,EAAE,YAAY,GAAG,+BAA+B;AAC3D;AAEO,SAAS,gBAAgB,aAA2B;AACzD,mBAAiB;AACjB,EAAAA,KAAI,KAAK,EAAE,YAAY,GAAG,qBAAqB;AAC/C,WAAS,iBAAiB,WAAW,cAAc,WAAW,KAAK,EAAE,OAAO,UAAU,CAAC;AACvF,EAAAA,KAAI,KAAK,EAAE,YAAY,GAAG,oBAAoB;AAChD;AAEO,SAAS,cAAsC;AACpD,QAAM,UAAe,WAAK,aAAa,cAAc;AACrD,MAAI,CAAI,eAAW,OAAO,EAAG,QAAO,CAAC;AACrC,QAAM,MAAM,KAAK,MAAS,iBAAa,SAAS,OAAO,CAAC;AACxD,SAAO,IAAI,gBAAgB,CAAC;AAC9B;AAEA,eAAsB,mBAAmB,aAAqD;AAC5F,MAAI;AACF,UAAMC,WAAU,cAAmB,WAAK,aAAa,cAAc,CAAC;AACpE,UAAM,WAAWA,SAAQ,QAAQ,WAAW;AAC5C,UAAM,MAAM,MAAM,OAAO;AAGzB,UAAM,UAAsC,IAAI,kBAAkB,IAAI;AACtE,QAAI,CAAC,WAAW,OAAO,QAAQ,kBAAkB,YAAY;AAC3D,MAAAD,KAAI,MAAM,EAAE,YAAY,GAAG,wEAAwE;AACnG,aAAO;AAAA,IACT;AACA,WAAO;AAAA,EACT,SAAS,KAAK;AACZ,IAAAA,KAAI,MAAM,EAAE,aAAa,IAAI,GAAG,uBAAuB;AACvD,IAAAA,KAAI,MAAM,EAAE,YAAY,GAAG,wCAAwC;AACnE,WAAO;AAAA,EACT;AACF;","names":["resolved","fs","path","log","require"]}
|