@remnic/server 9.3.638 → 9.3.639
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/README.md +15 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +388 -1
- package/dist/index.js.map +1 -1
- package/package.json +3 -3
package/README.md
CHANGED
|
@@ -30,6 +30,21 @@ npx --package @remnic/server remnic-server --help
|
|
|
30
30
|
npx --package @remnic/server remnic-server --port 4318
|
|
31
31
|
```
|
|
32
32
|
|
|
33
|
+
Run the container image:
|
|
34
|
+
|
|
35
|
+
```bash
|
|
36
|
+
docker run --rm \
|
|
37
|
+
-p 4318:4318 \
|
|
38
|
+
-v remnic-data:/data \
|
|
39
|
+
-e REMNIC_AUTH_TOKEN=change-me \
|
|
40
|
+
ghcr.io/joshuaswarren/remnic:latest
|
|
41
|
+
```
|
|
42
|
+
|
|
43
|
+
The image stores memory and runtime configuration under `/data`. The container
|
|
44
|
+
enables the admin console explicitly and serves it at `/engram/ui/`. Package
|
|
45
|
+
installs keep the console disabled unless `server.adminConsoleEnabled` or
|
|
46
|
+
`REMNIC_ADMIN_CONSOLE_ENABLED=true` is set with a public asset directory.
|
|
47
|
+
|
|
33
48
|
The package also ships the legacy `engram-server` binary for compatibility.
|
|
34
49
|
The bin wrappers are source-controlled so package managers can link them during
|
|
35
50
|
workspace installs; release builds verify that both targets have Node shebangs
|
package/dist/index.d.ts
CHANGED
|
@@ -24,6 +24,7 @@ interface ServerConfig {
|
|
|
24
24
|
maxBodyBytes?: number;
|
|
25
25
|
adminConsoleEnabled?: boolean;
|
|
26
26
|
adminConsolePublicDir?: string;
|
|
27
|
+
adminConsolePrefillToken?: boolean;
|
|
27
28
|
};
|
|
28
29
|
}
|
|
29
30
|
interface ParsedServerConfig {
|
|
@@ -34,6 +35,7 @@ interface ParsedServerConfig {
|
|
|
34
35
|
maxBodyBytes?: number;
|
|
35
36
|
adminConsoleEnabled: boolean;
|
|
36
37
|
adminConsolePublicDir?: string;
|
|
38
|
+
adminConsolePrefillToken: boolean;
|
|
37
39
|
}
|
|
38
40
|
declare function parseServerConfig(raw: Partial<ServerConfig["server"]>, options?: {
|
|
39
41
|
portSource?: string;
|
package/dist/index.js
CHANGED
|
@@ -55,7 +55,8 @@ function parseServerConfig(raw, options) {
|
|
|
55
55
|
principal: parseOptionalString(raw.principal, "server.principal"),
|
|
56
56
|
maxBodyBytes: parseOptionalPositiveInteger(raw.maxBodyBytes, "server.maxBodyBytes"),
|
|
57
57
|
adminConsoleEnabled: parseOptionalBoolean(raw.adminConsoleEnabled, "server.adminConsoleEnabled") ?? false,
|
|
58
|
-
adminConsolePublicDir: parseOptionalString(raw.adminConsolePublicDir, "server.adminConsolePublicDir")
|
|
58
|
+
adminConsolePublicDir: parseOptionalString(raw.adminConsolePublicDir, "server.adminConsolePublicDir"),
|
|
59
|
+
adminConsolePrefillToken: parseOptionalBoolean(raw.adminConsolePrefillToken, "server.adminConsolePrefillToken") ?? false
|
|
59
60
|
};
|
|
60
61
|
}
|
|
61
62
|
function resolveUserPath(value) {
|
|
@@ -129,9 +130,15 @@ function envOverrides() {
|
|
|
129
130
|
const port = readCompatEnv("REMNIC_PORT", "ENGRAM_PORT");
|
|
130
131
|
const host = readCompatEnv("REMNIC_HOST", "ENGRAM_HOST");
|
|
131
132
|
const authToken = readCompatEnv("REMNIC_AUTH_TOKEN", "ENGRAM_AUTH_TOKEN");
|
|
133
|
+
const adminConsoleEnabled = readCompatEnv("REMNIC_ADMIN_CONSOLE_ENABLED", "ENGRAM_ADMIN_CONSOLE_ENABLED");
|
|
134
|
+
const adminConsolePublicDir = readCompatEnv("REMNIC_ADMIN_CONSOLE_PUBLIC_DIR", "ENGRAM_ADMIN_CONSOLE_PUBLIC_DIR");
|
|
135
|
+
const adminConsolePrefillToken = readCompatEnv("REMNIC_ADMIN_CONSOLE_PREFILL_TOKEN", "ENGRAM_ADMIN_CONSOLE_PREFILL_TOKEN");
|
|
132
136
|
if (port) overrides.port = port;
|
|
133
137
|
if (host) overrides.host = host;
|
|
134
138
|
if (authToken) overrides.authToken = authToken;
|
|
139
|
+
if (adminConsoleEnabled) overrides.adminConsoleEnabled = adminConsoleEnabled;
|
|
140
|
+
if (adminConsolePublicDir) overrides.adminConsolePublicDir = adminConsolePublicDir;
|
|
141
|
+
if (adminConsolePrefillToken) overrides.adminConsolePrefillToken = adminConsolePrefillToken;
|
|
135
142
|
if (process.env.OPENAI_API_KEY) remnic.openaiApiKey = process.env.OPENAI_API_KEY;
|
|
136
143
|
const memoryDir = readCompatEnv("REMNIC_MEMORY_DIR", "ENGRAM_MEMORY_DIR");
|
|
137
144
|
if (memoryDir) remnic.memoryDir = memoryDir;
|
|
@@ -144,6 +151,382 @@ function mergeRemnicConfigForServer(fileRemnic, envRemnic) {
|
|
|
144
151
|
}
|
|
145
152
|
return { ...fileRemnic, ...effectiveEnvRemnic };
|
|
146
153
|
}
|
|
154
|
+
var WRITABLE_BOOLEAN_CONFIG_KEYS = /* @__PURE__ */ new Set([
|
|
155
|
+
"citationsAutoDetect",
|
|
156
|
+
"citationsEnabled",
|
|
157
|
+
"embeddingFallbackEnabled",
|
|
158
|
+
"enrichmentAutoOnCreate",
|
|
159
|
+
"enrichmentEnabled",
|
|
160
|
+
"feedbackEnabled",
|
|
161
|
+
"hostEmbeddingProviderEnabled",
|
|
162
|
+
"localLlmDisableThinking",
|
|
163
|
+
"localLlmEnabled",
|
|
164
|
+
"localLlmFallback",
|
|
165
|
+
"localLlmFastEnabled",
|
|
166
|
+
"memoryExtensionsEnabled",
|
|
167
|
+
"namespacesEnabled",
|
|
168
|
+
"qmdEnabled",
|
|
169
|
+
"queryExpansionEnabled",
|
|
170
|
+
"recallPlannerEnabled",
|
|
171
|
+
"recallPlannerLlmEnabled",
|
|
172
|
+
"recallPlannerTelemetryEnabled",
|
|
173
|
+
"rerankEnabled"
|
|
174
|
+
]);
|
|
175
|
+
var WRITABLE_STRING_CONFIG_KEYS = /* @__PURE__ */ new Set([
|
|
176
|
+
"embeddingFallbackModel",
|
|
177
|
+
"embeddingFallbackProvider",
|
|
178
|
+
"fastGatewayAgentId",
|
|
179
|
+
"gatewayAgentId",
|
|
180
|
+
"localLlmFastModel",
|
|
181
|
+
"localLlmModel",
|
|
182
|
+
"localLlmUrl",
|
|
183
|
+
"model",
|
|
184
|
+
"modelSource",
|
|
185
|
+
"openaiBaseUrl",
|
|
186
|
+
"qmdEmbedModel",
|
|
187
|
+
"qmdGenerateModel",
|
|
188
|
+
"qmdRerankModel",
|
|
189
|
+
"recallPlannerModel"
|
|
190
|
+
]);
|
|
191
|
+
function isWritableConfigKey(key) {
|
|
192
|
+
return WRITABLE_BOOLEAN_CONFIG_KEYS.has(key) || WRITABLE_STRING_CONFIG_KEYS.has(key);
|
|
193
|
+
}
|
|
194
|
+
function hasEnv(name) {
|
|
195
|
+
const value = process.env[name];
|
|
196
|
+
return typeof value === "string" && value.trim().length > 0;
|
|
197
|
+
}
|
|
198
|
+
function fileExists(candidate) {
|
|
199
|
+
try {
|
|
200
|
+
return fs.existsSync(expandTildePath(candidate));
|
|
201
|
+
} catch {
|
|
202
|
+
return false;
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
function canWriteConfigPath(configPath) {
|
|
206
|
+
try {
|
|
207
|
+
if (fs.existsSync(configPath)) {
|
|
208
|
+
fs.accessSync(configPath, fs.constants.W_OK);
|
|
209
|
+
return true;
|
|
210
|
+
}
|
|
211
|
+
fs.accessSync(path.dirname(configPath), fs.constants.W_OK);
|
|
212
|
+
return true;
|
|
213
|
+
} catch {
|
|
214
|
+
return false;
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
function writeConfigFileAtomically(configPath, data) {
|
|
218
|
+
fs.mkdirSync(path.dirname(configPath), { recursive: true });
|
|
219
|
+
const tmpPath = path.join(
|
|
220
|
+
path.dirname(configPath),
|
|
221
|
+
`.${path.basename(configPath)}.tmp-${process.pid}-${Date.now()}`
|
|
222
|
+
);
|
|
223
|
+
let completed = false;
|
|
224
|
+
try {
|
|
225
|
+
fs.writeFileSync(tmpPath, `${JSON.stringify(data, null, 2)}
|
|
226
|
+
`, {
|
|
227
|
+
encoding: "utf8",
|
|
228
|
+
flag: "wx",
|
|
229
|
+
mode: 384
|
|
230
|
+
});
|
|
231
|
+
fs.renameSync(tmpPath, configPath);
|
|
232
|
+
try {
|
|
233
|
+
fs.chmodSync(configPath, 384);
|
|
234
|
+
} catch {
|
|
235
|
+
}
|
|
236
|
+
completed = true;
|
|
237
|
+
} finally {
|
|
238
|
+
if (!completed) {
|
|
239
|
+
try {
|
|
240
|
+
fs.unlinkSync(tmpPath);
|
|
241
|
+
} catch {
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
function readJsonRecordIfPresent(configPath) {
|
|
247
|
+
if (!fs.existsSync(configPath)) return {};
|
|
248
|
+
const parsed = JSON.parse(fs.readFileSync(configPath, "utf8"));
|
|
249
|
+
if (!isPlainRecord(parsed)) {
|
|
250
|
+
throw new Error(`Invalid config file ${configPath}: top-level config must be a JSON object`);
|
|
251
|
+
}
|
|
252
|
+
return parsed;
|
|
253
|
+
}
|
|
254
|
+
function resolveEditableRemnicBlock(root) {
|
|
255
|
+
if (isPlainRecord(root.remnic)) return root.remnic;
|
|
256
|
+
if (isPlainRecord(root.engram)) return root.engram;
|
|
257
|
+
return root;
|
|
258
|
+
}
|
|
259
|
+
function normalizePatchValue(key, value) {
|
|
260
|
+
if (!isWritableConfigKey(key)) {
|
|
261
|
+
throw new Error(`Unsupported admin config key: ${key}`);
|
|
262
|
+
}
|
|
263
|
+
if (value === null) return null;
|
|
264
|
+
if (WRITABLE_BOOLEAN_CONFIG_KEYS.has(key)) {
|
|
265
|
+
if (typeof value === "boolean") return value;
|
|
266
|
+
throw new Error(`Invalid ${key}: expected boolean or null`);
|
|
267
|
+
}
|
|
268
|
+
if (typeof value !== "string") {
|
|
269
|
+
throw new Error(`Invalid ${key}: expected string or null`);
|
|
270
|
+
}
|
|
271
|
+
const normalized = value.trim();
|
|
272
|
+
if (key === "modelSource" && normalized !== "plugin" && normalized !== "gateway") {
|
|
273
|
+
throw new Error("Invalid modelSource: expected plugin or gateway");
|
|
274
|
+
}
|
|
275
|
+
if (key === "embeddingFallbackProvider" && normalized !== "auto" && normalized !== "openai" && normalized !== "local") {
|
|
276
|
+
throw new Error("Invalid embeddingFallbackProvider: expected auto, openai, or local");
|
|
277
|
+
}
|
|
278
|
+
return normalized;
|
|
279
|
+
}
|
|
280
|
+
function publicConfigValues(config, serverConfig) {
|
|
281
|
+
return {
|
|
282
|
+
adminConsoleEnabled: serverConfig.adminConsoleEnabled,
|
|
283
|
+
memoryDir: config.memoryDir,
|
|
284
|
+
model: config.model,
|
|
285
|
+
modelSource: config.modelSource,
|
|
286
|
+
gatewayAgentId: config.gatewayAgentId || null,
|
|
287
|
+
fastGatewayAgentId: config.fastGatewayAgentId || null,
|
|
288
|
+
localLlmEnabled: config.localLlmEnabled,
|
|
289
|
+
localLlmUrl: config.localLlmUrl || null,
|
|
290
|
+
localLlmModel: config.localLlmModel || null,
|
|
291
|
+
localLlmFastEnabled: config.localLlmFastEnabled,
|
|
292
|
+
localLlmFastModel: config.localLlmFastModel || null,
|
|
293
|
+
localLlmFallback: config.localLlmFallback,
|
|
294
|
+
localLlmDisableThinking: config.localLlmDisableThinking,
|
|
295
|
+
qmdEnabled: config.qmdEnabled,
|
|
296
|
+
qmdEmbedModel: config.qmdEmbedModel || null,
|
|
297
|
+
qmdRerankModel: config.qmdRerankModel || null,
|
|
298
|
+
qmdGenerateModel: config.qmdGenerateModel || null,
|
|
299
|
+
embeddingFallbackEnabled: config.embeddingFallbackEnabled,
|
|
300
|
+
embeddingFallbackProvider: config.embeddingFallbackProvider,
|
|
301
|
+
embeddingFallbackModel: config.embeddingFallbackModel || null,
|
|
302
|
+
hostEmbeddingProviderEnabled: config.hostEmbeddingProviderEnabled,
|
|
303
|
+
namespacesEnabled: config.namespacesEnabled,
|
|
304
|
+
recallPlannerEnabled: config.recallPlannerEnabled,
|
|
305
|
+
recallPlannerLlmEnabled: config.recallPlannerLlmEnabled,
|
|
306
|
+
recallPlannerModel: config.recallPlannerModel || null,
|
|
307
|
+
citationsEnabled: config.citationsEnabled,
|
|
308
|
+
citationsAutoDetect: config.citationsAutoDetect,
|
|
309
|
+
queryExpansionEnabled: config.queryExpansionEnabled,
|
|
310
|
+
rerankEnabled: config.rerankEnabled,
|
|
311
|
+
feedbackEnabled: config.feedbackEnabled,
|
|
312
|
+
memoryExtensionsEnabled: config.memoryExtensionsEnabled,
|
|
313
|
+
enrichmentEnabled: config.enrichmentEnabled,
|
|
314
|
+
enrichmentAutoOnCreate: config.enrichmentAutoOnCreate
|
|
315
|
+
};
|
|
316
|
+
}
|
|
317
|
+
function configuredModels(config) {
|
|
318
|
+
const models = /* @__PURE__ */ new Map();
|
|
319
|
+
const add = (id, provider, label, enabled, isDefault = false, source = "config") => {
|
|
320
|
+
const normalized = id?.trim();
|
|
321
|
+
if (!normalized) return;
|
|
322
|
+
const existing = models.get(`${provider}:${normalized}`);
|
|
323
|
+
models.set(`${provider}:${normalized}`, {
|
|
324
|
+
id: normalized,
|
|
325
|
+
provider,
|
|
326
|
+
label,
|
|
327
|
+
detected: existing?.detected ?? true,
|
|
328
|
+
enabled: existing?.enabled || enabled,
|
|
329
|
+
default: existing?.default || isDefault,
|
|
330
|
+
source
|
|
331
|
+
});
|
|
332
|
+
};
|
|
333
|
+
add(config.model, "openai", config.model, !isOpenaiApiKeyDisabled(config.openaiApiKey), config.modelSource === "plugin");
|
|
334
|
+
add(config.gatewayAgentId, "gateway", config.gatewayAgentId, config.modelSource === "gateway", config.modelSource === "gateway");
|
|
335
|
+
add(config.fastGatewayAgentId, "gateway", config.fastGatewayAgentId, config.modelSource === "gateway");
|
|
336
|
+
add(config.localLlmModel, "local", config.localLlmModel, config.localLlmEnabled, config.localLlmEnabled);
|
|
337
|
+
add(config.localLlmFastModel, "local", `${config.localLlmFastModel} (fast)`, config.localLlmFastEnabled);
|
|
338
|
+
add(config.embeddingFallbackModel, config.embeddingFallbackProvider, `${config.embeddingFallbackModel} (embedding fallback)`, config.embeddingFallbackEnabled);
|
|
339
|
+
add(config.qmdEmbedModel, "qmd", `${config.qmdEmbedModel} (embed)`, config.qmdEnabled);
|
|
340
|
+
add(config.qmdRerankModel, "qmd", `${config.qmdRerankModel} (rerank)`, config.qmdEnabled);
|
|
341
|
+
add(config.qmdGenerateModel, "qmd", `${config.qmdGenerateModel} (generate)`, config.qmdEnabled);
|
|
342
|
+
add(config.recallPlannerModel, "planner", `${config.recallPlannerModel} (planner)`, config.recallPlannerLlmEnabled);
|
|
343
|
+
return [...models.values()].sort((a, b) => `${a.provider}:${a.id}`.localeCompare(`${b.provider}:${b.id}`));
|
|
344
|
+
}
|
|
345
|
+
function isLikelyOllamaEndpoint(endpoint) {
|
|
346
|
+
return /(?:ollama|11434)/i.test(endpoint);
|
|
347
|
+
}
|
|
348
|
+
async function detectOllamaModels(baseUrl) {
|
|
349
|
+
const configuredEndpoint = baseUrl?.trim();
|
|
350
|
+
const envEndpoint = process.env.OLLAMA_HOST?.trim();
|
|
351
|
+
const endpoint = configuredEndpoint && isLikelyOllamaEndpoint(configuredEndpoint) ? configuredEndpoint : envEndpoint;
|
|
352
|
+
if (!endpoint) return [];
|
|
353
|
+
const controller = new AbortController();
|
|
354
|
+
const timer = setTimeout(() => controller.abort(), 900);
|
|
355
|
+
try {
|
|
356
|
+
const url = new URL("/api/tags", endpoint.endsWith("/") ? endpoint : `${endpoint}/`);
|
|
357
|
+
const response = await fetch(url, { signal: controller.signal });
|
|
358
|
+
if (!response.ok) return [];
|
|
359
|
+
const payload = await response.json();
|
|
360
|
+
return (payload.models ?? []).map((model) => typeof model.name === "string" ? model.name : typeof model.model === "string" ? model.model : "").filter((name) => name.length > 0).map((name) => ({
|
|
361
|
+
id: name,
|
|
362
|
+
label: name,
|
|
363
|
+
provider: "ollama",
|
|
364
|
+
detected: true,
|
|
365
|
+
enabled: true,
|
|
366
|
+
source: "ollama",
|
|
367
|
+
endpoint
|
|
368
|
+
}));
|
|
369
|
+
} catch {
|
|
370
|
+
return [];
|
|
371
|
+
} finally {
|
|
372
|
+
clearTimeout(timer);
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
function dashboardHarnesses(config) {
|
|
376
|
+
const openclawDetected = hasEnv("OPENCLAW_HOME") || hasEnv("OPENCLAW_WORKSPACE") || fileExists("~/.openclaw");
|
|
377
|
+
const codexDetected = hasEnv("CODEX_HOME") || fileExists("~/.codex/auth.json") || fileExists("~/.codex");
|
|
378
|
+
return [
|
|
379
|
+
{
|
|
380
|
+
id: "remnic-http",
|
|
381
|
+
label: "Remnic HTTP API",
|
|
382
|
+
detected: true,
|
|
383
|
+
enabled: true,
|
|
384
|
+
source: "server",
|
|
385
|
+
detail: "MCP and REST access server"
|
|
386
|
+
},
|
|
387
|
+
{
|
|
388
|
+
id: "openclaw",
|
|
389
|
+
label: "OpenClaw",
|
|
390
|
+
detected: openclawDetected,
|
|
391
|
+
enabled: config.modelSource === "gateway" || config.hostEmbeddingProviderEnabled,
|
|
392
|
+
source: openclawDetected ? "host" : "not detected",
|
|
393
|
+
detail: "Gateway and host adapters"
|
|
394
|
+
},
|
|
395
|
+
{
|
|
396
|
+
id: "codex",
|
|
397
|
+
label: "Codex",
|
|
398
|
+
detected: codexDetected,
|
|
399
|
+
enabled: config.citationsEnabled || config.citationsAutoDetect,
|
|
400
|
+
source: codexDetected ? "host" : "not detected",
|
|
401
|
+
detail: "Citation-aware adapter"
|
|
402
|
+
},
|
|
403
|
+
{
|
|
404
|
+
id: "qmd",
|
|
405
|
+
label: "QMD Search",
|
|
406
|
+
detected: Boolean(config.qmdPath) || config.qmdEnabled,
|
|
407
|
+
enabled: config.qmdEnabled,
|
|
408
|
+
source: config.qmdPath ? "qmdPath" : "config",
|
|
409
|
+
detail: config.qmdSearchStrategy
|
|
410
|
+
}
|
|
411
|
+
];
|
|
412
|
+
}
|
|
413
|
+
function dashboardProviders(config) {
|
|
414
|
+
const localLlmUrl = config.localLlmUrl || "";
|
|
415
|
+
const gatewayIds = [config.gatewayAgentId, config.fastGatewayAgentId].filter(Boolean).join(" ");
|
|
416
|
+
const openaiBaseUrl = process.env.OPENAI_BASE_URL || "";
|
|
417
|
+
const sageRouterDetected = hasEnv("SAGE_ROUTER_URL") || hasEnv("SAGE_ROUTER_HOST") || /sage[-_ ]?router/i.test(`${gatewayIds} ${openaiBaseUrl}`);
|
|
418
|
+
const ollamaDetected = hasEnv("OLLAMA_HOST") || /(?:ollama|11434)/i.test(localLlmUrl);
|
|
419
|
+
const localDetected = Boolean(localLlmUrl.trim());
|
|
420
|
+
return [
|
|
421
|
+
{
|
|
422
|
+
id: "openai",
|
|
423
|
+
label: "OpenAI",
|
|
424
|
+
detected: !isOpenaiApiKeyDisabled(config.openaiApiKey),
|
|
425
|
+
enabled: config.modelSource === "plugin" && !isOpenaiApiKeyDisabled(config.openaiApiKey),
|
|
426
|
+
source: !isOpenaiApiKeyDisabled(config.openaiApiKey) ? "config/env" : "disabled",
|
|
427
|
+
detail: config.model
|
|
428
|
+
},
|
|
429
|
+
{
|
|
430
|
+
id: "sage-router",
|
|
431
|
+
label: "Sage Router",
|
|
432
|
+
detected: sageRouterDetected,
|
|
433
|
+
enabled: config.modelSource === "gateway" && sageRouterDetected,
|
|
434
|
+
source: sageRouterDetected ? "gateway/env" : "not detected",
|
|
435
|
+
detail: config.gatewayAgentId || config.fastGatewayAgentId || openaiBaseUrl || "OpenAI-compatible provider router"
|
|
436
|
+
},
|
|
437
|
+
{
|
|
438
|
+
id: "ollama",
|
|
439
|
+
label: "Ollama",
|
|
440
|
+
detected: ollamaDetected,
|
|
441
|
+
enabled: config.localLlmEnabled && ollamaDetected,
|
|
442
|
+
source: ollamaDetected ? "localLlmUrl/OLLAMA_HOST" : "not detected",
|
|
443
|
+
detail: "Local or cloud-compatible Ollama endpoint"
|
|
444
|
+
},
|
|
445
|
+
{
|
|
446
|
+
id: "local-openai-compatible",
|
|
447
|
+
label: "Local OpenAI-compatible",
|
|
448
|
+
detected: localDetected && !ollamaDetected,
|
|
449
|
+
enabled: config.localLlmEnabled && localDetected && !ollamaDetected,
|
|
450
|
+
source: localDetected ? "localLlmUrl" : "not detected",
|
|
451
|
+
detail: localLlmUrl || "Local provider endpoint"
|
|
452
|
+
}
|
|
453
|
+
];
|
|
454
|
+
}
|
|
455
|
+
function dashboardFeatures(config) {
|
|
456
|
+
return [
|
|
457
|
+
["localLlmEnabled", "Local LLM", config.localLlmEnabled],
|
|
458
|
+
["localLlmFastEnabled", "Fast Local Tier", config.localLlmFastEnabled],
|
|
459
|
+
["qmdEnabled", "QMD Search", config.qmdEnabled],
|
|
460
|
+
["embeddingFallbackEnabled", "Embedding Fallback", config.embeddingFallbackEnabled],
|
|
461
|
+
["hostEmbeddingProviderEnabled", "Host Embeddings", config.hostEmbeddingProviderEnabled],
|
|
462
|
+
["namespacesEnabled", "Namespaces", config.namespacesEnabled],
|
|
463
|
+
["recallPlannerEnabled", "Recall Planner", config.recallPlannerEnabled],
|
|
464
|
+
["recallPlannerLlmEnabled", "Planner LLM", config.recallPlannerLlmEnabled],
|
|
465
|
+
["citationsEnabled", "Citations", config.citationsEnabled],
|
|
466
|
+
["citationsAutoDetect", "Citation Auto-detect", config.citationsAutoDetect],
|
|
467
|
+
["queryExpansionEnabled", "Query Expansion", config.queryExpansionEnabled],
|
|
468
|
+
["rerankEnabled", "Rerank", config.rerankEnabled],
|
|
469
|
+
["feedbackEnabled", "Feedback", config.feedbackEnabled],
|
|
470
|
+
["memoryExtensionsEnabled", "Memory Extensions", config.memoryExtensionsEnabled],
|
|
471
|
+
["enrichmentEnabled", "Entity Enrichment", config.enrichmentEnabled]
|
|
472
|
+
].map(([key, label, enabled]) => ({
|
|
473
|
+
key: String(key),
|
|
474
|
+
label: String(label),
|
|
475
|
+
enabled: enabled === true,
|
|
476
|
+
writable: WRITABLE_BOOLEAN_CONFIG_KEYS.has(String(key)),
|
|
477
|
+
restartRequired: true
|
|
478
|
+
}));
|
|
479
|
+
}
|
|
480
|
+
function createAdminControls(configPath, config, serverConfig) {
|
|
481
|
+
let restartRequired = false;
|
|
482
|
+
let displayConfig = config;
|
|
483
|
+
const status = async () => {
|
|
484
|
+
const models = configuredModels(displayConfig);
|
|
485
|
+
const ollamaModels = await detectOllamaModels(displayConfig.localLlmUrl);
|
|
486
|
+
const modelKeys = new Set(models.map((model) => `${model.provider}:${model.id}`));
|
|
487
|
+
for (const model of ollamaModels) {
|
|
488
|
+
if (!modelKeys.has(`${model.provider}:${model.id}`)) models.push(model);
|
|
489
|
+
}
|
|
490
|
+
return {
|
|
491
|
+
config: {
|
|
492
|
+
path: configPath,
|
|
493
|
+
exists: fs.existsSync(configPath),
|
|
494
|
+
writable: canWriteConfigPath(configPath),
|
|
495
|
+
restartRequired,
|
|
496
|
+
values: publicConfigValues(displayConfig, serverConfig)
|
|
497
|
+
},
|
|
498
|
+
harnesses: dashboardHarnesses(displayConfig),
|
|
499
|
+
providers: dashboardProviders(displayConfig),
|
|
500
|
+
models,
|
|
501
|
+
features: dashboardFeatures(displayConfig)
|
|
502
|
+
};
|
|
503
|
+
};
|
|
504
|
+
return {
|
|
505
|
+
status,
|
|
506
|
+
update: async (patch) => {
|
|
507
|
+
if (!isPlainRecord(patch)) {
|
|
508
|
+
throw new Error("Admin config patch must be an object");
|
|
509
|
+
}
|
|
510
|
+
const normalizedEntries = Object.entries(patch).map(([key, value]) => [key, normalizePatchValue(key, value)]);
|
|
511
|
+
if (normalizedEntries.length === 0) return status();
|
|
512
|
+
const raw = readJsonRecordIfPresent(configPath);
|
|
513
|
+
const target = resolveEditableRemnicBlock(raw);
|
|
514
|
+
for (const [key, value] of normalizedEntries) {
|
|
515
|
+
if (value === null) {
|
|
516
|
+
delete target[key];
|
|
517
|
+
} else {
|
|
518
|
+
target[key] = value;
|
|
519
|
+
}
|
|
520
|
+
}
|
|
521
|
+
const candidateRemnic = isPlainRecord(raw.remnic) ? raw.remnic : isPlainRecord(raw.engram) ? raw.engram : raw;
|
|
522
|
+
const nextDisplayConfig = parseConfig(candidateRemnic);
|
|
523
|
+
writeConfigFileAtomically(configPath, raw);
|
|
524
|
+
displayConfig = nextDisplayConfig;
|
|
525
|
+
restartRequired = true;
|
|
526
|
+
return status();
|
|
527
|
+
}
|
|
528
|
+
};
|
|
529
|
+
}
|
|
147
530
|
function abortableDelay(ms, signal) {
|
|
148
531
|
if (signal.aborted) return Promise.resolve();
|
|
149
532
|
return new Promise((resolve) => {
|
|
@@ -203,6 +586,8 @@ async function startServer(options) {
|
|
|
203
586
|
maxBodyBytes: parsedServerConfig.maxBodyBytes,
|
|
204
587
|
adminConsoleEnabled: parsedServerConfig.adminConsoleEnabled,
|
|
205
588
|
adminConsolePublicDir: parsedServerConfig.adminConsolePublicDir ? path.resolve(expandTildePath(parsedServerConfig.adminConsolePublicDir)) : void 0,
|
|
589
|
+
adminConsolePrefillToken: parsedServerConfig.adminConsolePrefillToken,
|
|
590
|
+
adminControls: parsedServerConfig.adminConsoleEnabled ? createAdminControls(resolvedConfigPath.path, config, parsedServerConfig) : void 0,
|
|
206
591
|
citationsEnabled: config.citationsEnabled,
|
|
207
592
|
citationsAutoDetect: config.citationsAutoDetect,
|
|
208
593
|
emitLegacyTools: config.emitLegacyTools
|
|
@@ -334,6 +719,8 @@ Environment:
|
|
|
334
719
|
REMNIC_PORT Server port (ENGRAM_PORT also supported)
|
|
335
720
|
REMNIC_HOST Bind address (ENGRAM_HOST also supported)
|
|
336
721
|
REMNIC_AUTH_TOKEN Auth bearer token (ENGRAM_AUTH_TOKEN also supported)
|
|
722
|
+
REMNIC_ADMIN_CONSOLE_PREFILL_TOKEN
|
|
723
|
+
Prefill admin UI with REMNIC_AUTH_TOKEN when true
|
|
337
724
|
REMNIC_MEMORY_DIR Override memory directory (ENGRAM_MEMORY_DIR also supported)
|
|
338
725
|
OPENAI_API_KEY OpenAI API key for extraction; ignored when config sets openaiApiKey=false
|
|
339
726
|
`);
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["/**\n * @remnic/server\n *\n * Standalone Remnic memory server.\n *\n * Loads config from `remnic.config.json` (or env vars), creates an Orchestrator,\n * and starts the HTTP access server with MCP endpoint — no OpenClaw required.\n *\n * Usage:\n * npx --package @remnic/server remnic-server\n * npx --package @remnic/server remnic-server --config ./my-remnic.json\n * npx --package @remnic/server remnic-server --port 4320\n */\n\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport { parseConfig, isOpenaiApiKeyDisabled, Orchestrator, EngramAccessService, EngramAccessHttpServer, initLogger, log, getAllValidTokens, getAllValidTokensCached, expandTildePath, type PluginConfig } from \"@remnic/core\";\n\n// ── Config loading ──────────────────────────────────────────────────────────\n\nexport interface ServerConfig {\n remnic: Record<string, unknown>;\n server: {\n host?: string;\n port?: unknown;\n authToken?: string;\n principal?: string;\n maxBodyBytes?: number;\n adminConsoleEnabled?: boolean;\n adminConsolePublicDir?: string;\n };\n}\n\nfunction readCompatEnv(primary: string, legacy: string): string | undefined {\n return process.env[primary] ?? process.env[legacy];\n}\n\nfunction parseServerPort(value: unknown, source: string): number {\n const port = typeof value === \"string\" ? Number(value.trim()) : value;\n if (\n typeof port !== \"number\" ||\n !Number.isInteger(port) ||\n port < 1 ||\n port > 65535\n ) {\n throw new Error(`Invalid ${source}: expected an integer port from 1 to 65535`);\n }\n return port;\n}\n\nfunction parseOptionalString(value: unknown, source: string): string | undefined {\n if (value === undefined) return undefined;\n if (typeof value !== \"string\") {\n throw new Error(`Invalid ${source}: expected a string`);\n }\n return value;\n}\n\nfunction parseOptionalNonEmptyString(value: unknown, source: string): string | undefined {\n const parsed = parseOptionalString(value, source);\n if (parsed === undefined) return undefined;\n if (parsed.trim() === \"\") {\n throw new Error(`Invalid ${source}: expected a non-empty string`);\n }\n return parsed;\n}\n\nfunction parseOptionalPositiveInteger(value: unknown, source: string): number | undefined {\n if (value === undefined) return undefined;\n const parsed = typeof value === \"string\" ? Number(value.trim()) : value;\n if (\n typeof parsed !== \"number\" ||\n !Number.isInteger(parsed) ||\n parsed < 1\n ) {\n throw new Error(`Invalid ${source}: expected a positive integer`);\n }\n return parsed;\n}\n\nfunction parseOptionalBoolean(value: unknown, source: string): boolean | undefined {\n if (value === undefined) return undefined;\n if (typeof value === \"boolean\") return value;\n if (typeof value === \"string\") {\n const normalized = value.trim().toLowerCase();\n if ([\"true\", \"1\", \"yes\", \"on\"].includes(normalized)) return true;\n if ([\"false\", \"0\", \"no\", \"off\"].includes(normalized)) return false;\n }\n throw new Error(`Invalid ${source}: expected a boolean`);\n}\n\nexport interface ParsedServerConfig {\n host: string;\n port: number;\n authToken?: string;\n principal?: string;\n maxBodyBytes?: number;\n adminConsoleEnabled: boolean;\n adminConsolePublicDir?: string;\n}\n\nexport function parseServerConfig(\n raw: Partial<ServerConfig[\"server\"]>,\n options?: { portSource?: string },\n): ParsedServerConfig {\n return {\n host: parseOptionalNonEmptyString(raw.host, \"server.host\") ?? \"127.0.0.1\",\n port: raw.port === undefined\n ? 4318\n : parseServerPort(raw.port, options?.portSource ?? \"server.port\"),\n authToken: parseOptionalString(raw.authToken, \"server.authToken\"),\n principal: parseOptionalString(raw.principal, \"server.principal\"),\n maxBodyBytes: parseOptionalPositiveInteger(raw.maxBodyBytes, \"server.maxBodyBytes\"),\n adminConsoleEnabled: parseOptionalBoolean(raw.adminConsoleEnabled, \"server.adminConsoleEnabled\") ?? false,\n adminConsolePublicDir: parseOptionalString(raw.adminConsolePublicDir, \"server.adminConsolePublicDir\"),\n };\n}\n\ninterface ResolvedConfigPath {\n path: string;\n explicit: boolean;\n source: string;\n}\n\nfunction resolveUserPath(value: string): string {\n return path.resolve(expandTildePath(value));\n}\n\nfunction resolveConfigPath(cliPath?: string): ResolvedConfigPath {\n if (cliPath) {\n return { path: resolveUserPath(cliPath), explicit: true, source: \"--config\" };\n }\n\n const envPath = readCompatEnv(\"REMNIC_CONFIG_PATH\", \"ENGRAM_CONFIG_PATH\");\n if (envPath) {\n return { path: resolveUserPath(envPath), explicit: true, source: \"REMNIC_CONFIG_PATH/ENGRAM_CONFIG_PATH\" };\n }\n\n const homeDir = process.env.HOME ?? \"~\";\n const candidates = [\n path.join(process.cwd(), \"remnic.config.json\"),\n path.join(process.cwd(), \"engram.config.json\"),\n path.join(homeDir, \".config\", \"remnic\", \"config.json\"),\n path.join(homeDir, \".config\", \"engram\", \"config.json\"),\n ];\n for (const candidate of candidates) {\n if (fs.existsSync(candidate) && fs.statSync(candidate).isFile()) {\n return { path: candidate, explicit: false, source: \"auto-discovery\" };\n }\n }\n\n return { path: path.join(homeDir, \".config\", \"remnic\", \"config.json\"), explicit: false, source: \"auto-discovery\" };\n}\n\nfunction isPlainRecord(value: unknown): value is Record<string, unknown> {\n return !!value && typeof value === \"object\" && !Array.isArray(value);\n}\n\nfunction requirePlainConfigBlock(\n raw: Record<string, unknown>,\n key: \"remnic\" | \"engram\" | \"server\",\n configPath: string,\n): Record<string, unknown> | undefined {\n const value = raw[key];\n if (value === undefined) return undefined;\n if (!isPlainRecord(value)) {\n throw new Error(`Invalid config file ${configPath}: ${key} must be a JSON object`);\n }\n return value;\n}\n\nexport function loadConfigFile(configPath: string): ServerConfig {\n const raw = JSON.parse(fs.readFileSync(configPath, \"utf8\"));\n if (!isPlainRecord(raw)) {\n throw new Error(`Invalid config file ${configPath}: top-level config must be a JSON object`);\n }\n const remnic = requirePlainConfigBlock(raw, \"remnic\", configPath);\n const engram = requirePlainConfigBlock(raw, \"engram\", configPath);\n const server = requirePlainConfigBlock(raw, \"server\", configPath);\n return {\n remnic: remnic ?? engram ?? raw,\n server: server ?? {},\n };\n}\n\nfunction loadResolvedConfig(resolved: ResolvedConfigPath): ServerConfig {\n if (!fs.existsSync(resolved.path)) {\n if (resolved.explicit) {\n throw new Error(`Config file from ${resolved.source} not found: ${resolved.path}`);\n }\n return { remnic: {}, server: {} };\n }\n\n const stat = fs.statSync(resolved.path);\n if (!stat.isFile()) {\n if (!resolved.explicit) {\n return { remnic: {}, server: {} };\n }\n throw new Error(`Config file from ${resolved.source} is not a regular file: ${resolved.path}`);\n }\n\n return loadConfigFile(resolved.path);\n}\n\nfunction envOverrides(): Partial<ServerConfig[\"server\"]> & { remnic?: Record<string, unknown> } {\n const overrides: Record<string, unknown> = {};\n const remnic: Record<string, unknown> = {};\n\n const port = readCompatEnv(\"REMNIC_PORT\", \"ENGRAM_PORT\");\n const host = readCompatEnv(\"REMNIC_HOST\", \"ENGRAM_HOST\");\n const authToken = readCompatEnv(\"REMNIC_AUTH_TOKEN\", \"ENGRAM_AUTH_TOKEN\");\n if (port) overrides.port = port;\n if (host) overrides.host = host;\n if (authToken) overrides.authToken = authToken;\n\n if (process.env.OPENAI_API_KEY) remnic.openaiApiKey = process.env.OPENAI_API_KEY;\n const memoryDir = readCompatEnv(\"REMNIC_MEMORY_DIR\", \"ENGRAM_MEMORY_DIR\");\n if (memoryDir) remnic.memoryDir = memoryDir;\n\n return { ...overrides, ...(Object.keys(remnic).length > 0 ? { remnic } : {}) };\n}\n\nexport function mergeRemnicConfigForServer(\n fileRemnic: Record<string, unknown>,\n envRemnic: Record<string, unknown> | undefined,\n): Record<string, unknown> {\n const effectiveEnvRemnic = { ...(envRemnic ?? {}) };\n if (isOpenaiApiKeyDisabled(fileRemnic.openaiApiKey)) {\n // A local/gateway-only deployment can explicitly disable the direct\n // OpenAI client. Preserve that opt-out even when the process has a\n // global OPENAI_API_KEY for unrelated tools.\n delete effectiveEnvRemnic.openaiApiKey;\n }\n return { ...fileRemnic, ...effectiveEnvRemnic };\n}\n\n// ── Helpers ─────────────────────────────────────────────────────────────────\n\n/**\n * Like `setTimeout` wrapped in a Promise, but respects an `AbortSignal`.\n * Resolves immediately (without throwing) when the signal fires so the\n * caller can check `signal.aborted` and exit cleanly.\n */\nfunction abortableDelay(ms: number, signal: AbortSignal): Promise<void> {\n if (signal.aborted) return Promise.resolve();\n return new Promise<void>((resolve) => {\n const timer = setTimeout(resolve, ms);\n const onAbort = () => {\n clearTimeout(timer);\n resolve();\n };\n signal.addEventListener(\"abort\", onAbort, { once: true });\n });\n}\n\nasync function cleanupFailedStartup(\n orchestrator: Orchestrator,\n httpServer: EngramAccessHttpServer,\n): Promise<void> {\n try {\n await httpServer.stop();\n } catch (err) {\n log.warn(`HTTP startup failure cleanup could not stop server: ${err}`);\n }\n\n try {\n await orchestrator.destroy();\n } catch (err) {\n log.warn(`HTTP startup failure cleanup could not destroy orchestrator: ${err}`);\n }\n}\n\n// ── Server startup ──────────────────────────────────────────────────────────\n\nexport interface ServerResult {\n config: PluginConfig;\n service: EngramAccessService;\n httpServer: EngramAccessHttpServer;\n host: string;\n port: number;\n /** Stop HTTP, cancel startup work, abort deferred init, and destroy the orchestrator. */\n stop: () => Promise<void>;\n /** Cancel any pending startup-sync retry timers. Called automatically on shutdown. */\n cancelStartupSync: () => void;\n /** Abort deferred orchestrator initialization (QMD sync, warmup, cache). */\n abortDeferredInit: () => void;\n}\n\nexport async function startServer(options?: {\n configPath?: string;\n host?: string;\n port?: number;\n authToken?: string;\n}): Promise<ServerResult> {\n initLogger();\n\n const resolvedConfigPath = resolveConfigPath(options?.configPath);\n const fileConfig = loadResolvedConfig(resolvedConfigPath);\n\n const env = envOverrides();\n const { remnic: envRemnic, ...envServer } = env;\n\n // Merge: file < env < cli flags\n const remnicConfig = mergeRemnicConfigForServer(fileConfig.remnic, envRemnic);\n const cliServerConfig: Partial<ServerConfig[\"server\"]> = {};\n if (options?.host !== undefined) cliServerConfig.host = options.host;\n if (options?.port !== undefined) cliServerConfig.port = parseServerPort(options.port, \"options.port\");\n if (options?.authToken !== undefined) cliServerConfig.authToken = options.authToken;\n\n const serverConfig = {\n ...fileConfig.server,\n ...envServer,\n ...cliServerConfig,\n };\n const portSource = cliServerConfig.port !== undefined\n ? \"options.port\"\n : envServer.port !== undefined\n ? \"REMNIC_PORT/ENGRAM_PORT\"\n : \"server.port\";\n const parsedServerConfig = parseServerConfig(serverConfig, { portSource });\n\n const config = parseConfig(remnicConfig);\n const orchestrator = new Orchestrator(config);\n await orchestrator.initialize();\n\n // Start the HTTP server immediately so health checks, MCP handshakes,\n // and liveness probes can connect while deferred init is still running.\n const service = new EngramAccessService(orchestrator);\n\n const authToken = parsedServerConfig.authToken ?? readCompatEnv(\"REMNIC_AUTH_TOKEN\", \"ENGRAM_AUTH_TOKEN\") ?? \"\";\n\n // Connector tokens are loaded dynamically per request via authTokensGetter\n // so that token generate/revoke takes effect without server restart\n if (!authToken && getAllValidTokens().length === 0) {\n log.warn(\"No auth token set — server will reject all requests. Set REMNIC_AUTH_TOKEN, server.authToken in config, or generate tokens with 'remnic token generate'.\");\n }\n\n const httpServer = new EngramAccessHttpServer({\n service,\n host: parsedServerConfig.host,\n port: parsedServerConfig.port,\n authToken: authToken || undefined,\n authTokensGetter: () => getAllValidTokensCached(),\n principal: parsedServerConfig.principal,\n maxBodyBytes: parsedServerConfig.maxBodyBytes,\n adminConsoleEnabled: parsedServerConfig.adminConsoleEnabled,\n adminConsolePublicDir: parsedServerConfig.adminConsolePublicDir\n ? path.resolve(expandTildePath(parsedServerConfig.adminConsolePublicDir))\n : undefined,\n citationsEnabled: config.citationsEnabled,\n citationsAutoDetect: config.citationsAutoDetect,\n emitLegacyTools: config.emitLegacyTools,\n });\n\n let host: string;\n let port: number;\n try {\n ({ host, port } = await httpServer.start());\n } catch (err) {\n await cleanupFailedStartup(orchestrator, httpServer);\n throw err;\n }\n\n // Fire-and-forget: wait for deferred init (QMD probe, collection setup,\n // warmup) then check QMD availability and retry if needed. This does NOT\n // block the server listener — connections are accepted immediately above.\n // An AbortController allows the shutdown handler to cancel pending retries.\n const startupSyncAbort = new AbortController();\n\n // Wrap httpServer.stop() so that existing callers also get full lifecycle\n // cleanup: retry timers, deferred init, HTTP listener, and orchestrator.\n const originalStop = httpServer.stop.bind(httpServer);\n let stopPromise: Promise<void> | undefined;\n const stop = async (): Promise<void> => {\n if (stopPromise) return stopPromise;\n stopPromise = (async () => {\n startupSyncAbort.abort();\n orchestrator.abortDeferredInit();\n try {\n await originalStop();\n } finally {\n await orchestrator.destroy();\n }\n })();\n return stopPromise;\n };\n httpServer.stop = stop;\n\n orchestrator.deferredReady.then(() => {\n if (startupSyncAbort.signal.aborted) {\n log.debug(\"QMD startup-sync: cancelled before deferred init completed\");\n return;\n }\n\n // Skip retries when search is explicitly disabled via config or when the\n // orchestrator already resolved to a noop backend (e.g. missing collection\n // detected during deferredInitialize). Both cases mean no sync should ever\n // run; scheduling retries would create misleading operational noise and\n // unnecessary background work on every server start.\n if (!config.qmdEnabled || orchestrator.qmd.debugStatus() === \"backend=noop\") {\n log.debug(\"QMD startup-sync: search disabled or noop backend, skipping retries\");\n return;\n }\n\n // Retry when either: (a) QMD is not available yet (cold-start race), or\n // (b) QMD is available but the deferred init sync step failed silently\n // (e.g., update errors swallowed by backend, throttle skip, transient\n // network failure). Without (b), the daemon permanently serves stale\n // recall after a failed sync despite healthy QMD probe.\n const needsRetry = !orchestrator.qmd.isAvailable() || !orchestrator.deferredSyncSucceeded;\n if (!needsRetry) {\n log.debug(\"QMD startup-sync: deferred init completed successfully, no retries needed\");\n return;\n }\n\n const RETRY_DELAYS_MS = [5_000, 15_000, 30_000, 60_000, 120_000];\n if (startupSyncAbort.signal.aborted) {\n log.debug(\"QMD startup-sync retry: cancelled before retry task started\");\n return;\n }\n (async () => {\n for (const delay of RETRY_DELAYS_MS) {\n await abortableDelay(delay, startupSyncAbort.signal);\n\n if (startupSyncAbort.signal.aborted) {\n log.debug(\"QMD startup-sync retry: cancelled by shutdown\");\n return;\n }\n\n const synced = await orchestrator.startupSearchSync(startupSyncAbort.signal);\n if (!synced) {\n if (orchestrator.qmd.debugStatus() === \"backend=noop\") {\n log.debug(\"QMD startup-sync retry: search intentionally disabled; stopping retries\");\n return;\n }\n log.debug(`QMD startup-sync retry: not available yet (next retry in ${RETRY_DELAYS_MS[RETRY_DELAYS_MS.indexOf(delay) + 1] ?? \"n/a\"}ms)`);\n continue;\n }\n\n return; // sync succeeded, stop retrying\n }\n\n log.warn(\"QMD startup-sync retry: exhausted all retries; search index may be stale\");\n })().catch((err: unknown) => {\n log.warn(`QMD startup-sync retry: unexpected error: ${err}`);\n });\n }).catch((err: unknown) => {\n log.warn(`Deferred init error: ${err}`);\n });\n\n return { config, service, httpServer, host, port, stop, cancelStartupSync: () => startupSyncAbort.abort(), abortDeferredInit: () => orchestrator.abortDeferredInit() };\n}\n\n// ── CLI entry point ──────────────────────────────────────────────────────────\n\nconst BOOLEAN_CLI_OPTIONS = new Set([\"help\"]);\nconst VALUE_CLI_OPTIONS = new Set([\"config\", \"host\", \"port\", \"auth-token\"]);\n\nfunction parseCliArgs(argv: string[]): Record<string, string | undefined> {\n const args: Record<string, string | undefined> = {};\n for (let i = 0; i < argv.length; i++) {\n const token = argv[i];\n if (token === \"-h\") {\n args.help = \"true\";\n continue;\n }\n\n if (token.startsWith(\"--\")) {\n const [key, inlineValue] = token.slice(2).split(/=(.*)/s, 2);\n if (!key) {\n throw new Error(`Invalid option ${token}`);\n }\n\n if (BOOLEAN_CLI_OPTIONS.has(key)) {\n if (inlineValue !== undefined) {\n throw new Error(`Option --${key} does not accept a value`);\n }\n args[key] = \"true\";\n continue;\n }\n\n if (!VALUE_CLI_OPTIONS.has(key)) {\n throw new Error(`Unknown option --${key}`);\n }\n\n const value = inlineValue ?? argv[i + 1];\n if (\n value === undefined ||\n (inlineValue === undefined && value.startsWith(\"--\")) ||\n value.trim() === \"\"\n ) {\n throw new Error(`Missing value for --${key}`);\n }\n\n args[key] = value;\n if (inlineValue === undefined) i++;\n }\n }\n return args;\n}\n\nexport async function cliMain(argv: string[] = process.argv.slice(2)): Promise<void> {\n const args = parseCliArgs(argv);\n\n if (args.help) {\n console.log(`\nremnic-server — Standalone Remnic memory server\n\nUsage:\n remnic-server [options]\n\nOptions:\n --config <path> Path to config file (default: remnic.config.json)\n --host <addr> Bind address (default: 127.0.0.1)\n --port <number> Port number (default: 4318)\n --auth-token <tok> Bearer token for auth (or set REMNIC_AUTH_TOKEN)\n --help Show this help\n\nEnvironment:\n REMNIC_CONFIG_PATH Config file path (ENGRAM_CONFIG_PATH also supported)\n REMNIC_PORT Server port (ENGRAM_PORT also supported)\n REMNIC_HOST Bind address (ENGRAM_HOST also supported)\n REMNIC_AUTH_TOKEN Auth bearer token (ENGRAM_AUTH_TOKEN also supported)\n REMNIC_MEMORY_DIR Override memory directory (ENGRAM_MEMORY_DIR also supported)\n OPENAI_API_KEY OpenAI API key for extraction; ignored when config sets openaiApiKey=false\n`);\n process.exit(0);\n }\n\n const result = await startServer({\n configPath: args.config,\n host: args.host,\n port: args.port === undefined ? undefined : parseServerPort(args.port, \"--port\"),\n authToken: args[\"auth-token\"],\n });\n\n console.log(`Remnic server listening on http://${result.host}:${result.port}`);\n\n // Graceful shutdown\n const shutdown = async (signal: string) => {\n console.log(`\\nReceived ${signal}, shutting down...`);\n await result.stop();\n process.exit(0);\n };\n\n process.on(\"SIGINT\", () => shutdown(\"SIGINT\"));\n process.on(\"SIGTERM\", () => shutdown(\"SIGTERM\"));\n}\n\n// Auto-run when executed directly\n// Matches direct execution of `node .../remnic-server/dist/index.js` or\n// `node .../remnic-server/src/index.ts`. Package command names are handled by\n// the bin wrappers in ../bin so importing this module cannot start twice.\nif (\n process.argv[1] &&\n /(?:remnic-server|engram-server)[\\\\/](?:dist|src)[\\\\/]index\\.[jt]s$/.test(process.argv[1])\n) {\n cliMain().catch((err) => {\n process.stderr.write(`Fatal: ${err instanceof Error ? err.message : String(err)}\\n`);\n process.exit(1);\n });\n}\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["/**\n * @remnic/server\n *\n * Standalone Remnic memory server.\n *\n * Loads config from `remnic.config.json` (or env vars), creates an Orchestrator,\n * and starts the HTTP access server with MCP endpoint — no OpenClaw required.\n *\n * Usage:\n * npx --package @remnic/server remnic-server\n * npx --package @remnic/server remnic-server --config ./my-remnic.json\n * npx --package @remnic/server remnic-server --port 4320\n */\n\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport { parseConfig, isOpenaiApiKeyDisabled, Orchestrator, EngramAccessService, EngramAccessHttpServer, initLogger, log, getAllValidTokens, getAllValidTokensCached, expandTildePath, type PluginConfig, type RemnicAdminControls, type RemnicAdminDashboardStatus, type RemnicAdminModelOption, type RemnicAdminConfigPatch } from \"@remnic/core\";\n\n// ── Config loading ──────────────────────────────────────────────────────────\n\nexport interface ServerConfig {\n remnic: Record<string, unknown>;\n server: {\n host?: string;\n port?: unknown;\n authToken?: string;\n principal?: string;\n maxBodyBytes?: number;\n adminConsoleEnabled?: boolean;\n adminConsolePublicDir?: string;\n adminConsolePrefillToken?: boolean;\n };\n}\n\nfunction readCompatEnv(primary: string, legacy: string): string | undefined {\n return process.env[primary] ?? process.env[legacy];\n}\n\nfunction parseServerPort(value: unknown, source: string): number {\n const port = typeof value === \"string\" ? Number(value.trim()) : value;\n if (\n typeof port !== \"number\" ||\n !Number.isInteger(port) ||\n port < 1 ||\n port > 65535\n ) {\n throw new Error(`Invalid ${source}: expected an integer port from 1 to 65535`);\n }\n return port;\n}\n\nfunction parseOptionalString(value: unknown, source: string): string | undefined {\n if (value === undefined) return undefined;\n if (typeof value !== \"string\") {\n throw new Error(`Invalid ${source}: expected a string`);\n }\n return value;\n}\n\nfunction parseOptionalNonEmptyString(value: unknown, source: string): string | undefined {\n const parsed = parseOptionalString(value, source);\n if (parsed === undefined) return undefined;\n if (parsed.trim() === \"\") {\n throw new Error(`Invalid ${source}: expected a non-empty string`);\n }\n return parsed;\n}\n\nfunction parseOptionalPositiveInteger(value: unknown, source: string): number | undefined {\n if (value === undefined) return undefined;\n const parsed = typeof value === \"string\" ? Number(value.trim()) : value;\n if (\n typeof parsed !== \"number\" ||\n !Number.isInteger(parsed) ||\n parsed < 1\n ) {\n throw new Error(`Invalid ${source}: expected a positive integer`);\n }\n return parsed;\n}\n\nfunction parseOptionalBoolean(value: unknown, source: string): boolean | undefined {\n if (value === undefined) return undefined;\n if (typeof value === \"boolean\") return value;\n if (typeof value === \"string\") {\n const normalized = value.trim().toLowerCase();\n if ([\"true\", \"1\", \"yes\", \"on\"].includes(normalized)) return true;\n if ([\"false\", \"0\", \"no\", \"off\"].includes(normalized)) return false;\n }\n throw new Error(`Invalid ${source}: expected a boolean`);\n}\n\nexport interface ParsedServerConfig {\n host: string;\n port: number;\n authToken?: string;\n principal?: string;\n maxBodyBytes?: number;\n adminConsoleEnabled: boolean;\n adminConsolePublicDir?: string;\n adminConsolePrefillToken: boolean;\n}\n\nexport function parseServerConfig(\n raw: Partial<ServerConfig[\"server\"]>,\n options?: { portSource?: string },\n): ParsedServerConfig {\n return {\n host: parseOptionalNonEmptyString(raw.host, \"server.host\") ?? \"127.0.0.1\",\n port: raw.port === undefined\n ? 4318\n : parseServerPort(raw.port, options?.portSource ?? \"server.port\"),\n authToken: parseOptionalString(raw.authToken, \"server.authToken\"),\n principal: parseOptionalString(raw.principal, \"server.principal\"),\n maxBodyBytes: parseOptionalPositiveInteger(raw.maxBodyBytes, \"server.maxBodyBytes\"),\n adminConsoleEnabled: parseOptionalBoolean(raw.adminConsoleEnabled, \"server.adminConsoleEnabled\") ?? false,\n adminConsolePublicDir: parseOptionalString(raw.adminConsolePublicDir, \"server.adminConsolePublicDir\"),\n adminConsolePrefillToken: parseOptionalBoolean(raw.adminConsolePrefillToken, \"server.adminConsolePrefillToken\") ?? false,\n };\n}\n\ninterface ResolvedConfigPath {\n path: string;\n explicit: boolean;\n source: string;\n}\n\nfunction resolveUserPath(value: string): string {\n return path.resolve(expandTildePath(value));\n}\n\nfunction resolveConfigPath(cliPath?: string): ResolvedConfigPath {\n if (cliPath) {\n return { path: resolveUserPath(cliPath), explicit: true, source: \"--config\" };\n }\n\n const envPath = readCompatEnv(\"REMNIC_CONFIG_PATH\", \"ENGRAM_CONFIG_PATH\");\n if (envPath) {\n return { path: resolveUserPath(envPath), explicit: true, source: \"REMNIC_CONFIG_PATH/ENGRAM_CONFIG_PATH\" };\n }\n\n const homeDir = process.env.HOME ?? \"~\";\n const candidates = [\n path.join(process.cwd(), \"remnic.config.json\"),\n path.join(process.cwd(), \"engram.config.json\"),\n path.join(homeDir, \".config\", \"remnic\", \"config.json\"),\n path.join(homeDir, \".config\", \"engram\", \"config.json\"),\n ];\n for (const candidate of candidates) {\n if (fs.existsSync(candidate) && fs.statSync(candidate).isFile()) {\n return { path: candidate, explicit: false, source: \"auto-discovery\" };\n }\n }\n\n return { path: path.join(homeDir, \".config\", \"remnic\", \"config.json\"), explicit: false, source: \"auto-discovery\" };\n}\n\nfunction isPlainRecord(value: unknown): value is Record<string, unknown> {\n return !!value && typeof value === \"object\" && !Array.isArray(value);\n}\n\nfunction requirePlainConfigBlock(\n raw: Record<string, unknown>,\n key: \"remnic\" | \"engram\" | \"server\",\n configPath: string,\n): Record<string, unknown> | undefined {\n const value = raw[key];\n if (value === undefined) return undefined;\n if (!isPlainRecord(value)) {\n throw new Error(`Invalid config file ${configPath}: ${key} must be a JSON object`);\n }\n return value;\n}\n\nexport function loadConfigFile(configPath: string): ServerConfig {\n const raw = JSON.parse(fs.readFileSync(configPath, \"utf8\"));\n if (!isPlainRecord(raw)) {\n throw new Error(`Invalid config file ${configPath}: top-level config must be a JSON object`);\n }\n const remnic = requirePlainConfigBlock(raw, \"remnic\", configPath);\n const engram = requirePlainConfigBlock(raw, \"engram\", configPath);\n const server = requirePlainConfigBlock(raw, \"server\", configPath);\n return {\n remnic: remnic ?? engram ?? raw,\n server: server ?? {},\n };\n}\n\nfunction loadResolvedConfig(resolved: ResolvedConfigPath): ServerConfig {\n if (!fs.existsSync(resolved.path)) {\n if (resolved.explicit) {\n throw new Error(`Config file from ${resolved.source} not found: ${resolved.path}`);\n }\n return { remnic: {}, server: {} };\n }\n\n const stat = fs.statSync(resolved.path);\n if (!stat.isFile()) {\n if (!resolved.explicit) {\n return { remnic: {}, server: {} };\n }\n throw new Error(`Config file from ${resolved.source} is not a regular file: ${resolved.path}`);\n }\n\n return loadConfigFile(resolved.path);\n}\n\nfunction envOverrides(): Partial<ServerConfig[\"server\"]> & { remnic?: Record<string, unknown> } {\n const overrides: Record<string, unknown> = {};\n const remnic: Record<string, unknown> = {};\n\n const port = readCompatEnv(\"REMNIC_PORT\", \"ENGRAM_PORT\");\n const host = readCompatEnv(\"REMNIC_HOST\", \"ENGRAM_HOST\");\n const authToken = readCompatEnv(\"REMNIC_AUTH_TOKEN\", \"ENGRAM_AUTH_TOKEN\");\n const adminConsoleEnabled = readCompatEnv(\"REMNIC_ADMIN_CONSOLE_ENABLED\", \"ENGRAM_ADMIN_CONSOLE_ENABLED\");\n const adminConsolePublicDir = readCompatEnv(\"REMNIC_ADMIN_CONSOLE_PUBLIC_DIR\", \"ENGRAM_ADMIN_CONSOLE_PUBLIC_DIR\");\n const adminConsolePrefillToken = readCompatEnv(\"REMNIC_ADMIN_CONSOLE_PREFILL_TOKEN\", \"ENGRAM_ADMIN_CONSOLE_PREFILL_TOKEN\");\n if (port) overrides.port = port;\n if (host) overrides.host = host;\n if (authToken) overrides.authToken = authToken;\n if (adminConsoleEnabled) overrides.adminConsoleEnabled = adminConsoleEnabled;\n if (adminConsolePublicDir) overrides.adminConsolePublicDir = adminConsolePublicDir;\n if (adminConsolePrefillToken) overrides.adminConsolePrefillToken = adminConsolePrefillToken;\n\n if (process.env.OPENAI_API_KEY) remnic.openaiApiKey = process.env.OPENAI_API_KEY;\n const memoryDir = readCompatEnv(\"REMNIC_MEMORY_DIR\", \"ENGRAM_MEMORY_DIR\");\n if (memoryDir) remnic.memoryDir = memoryDir;\n\n return { ...overrides, ...(Object.keys(remnic).length > 0 ? { remnic } : {}) };\n}\n\nexport function mergeRemnicConfigForServer(\n fileRemnic: Record<string, unknown>,\n envRemnic: Record<string, unknown> | undefined,\n): Record<string, unknown> {\n const effectiveEnvRemnic = { ...(envRemnic ?? {}) };\n if (isOpenaiApiKeyDisabled(fileRemnic.openaiApiKey)) {\n // A local/gateway-only deployment can explicitly disable the direct\n // OpenAI client. Preserve that opt-out even when the process has a\n // global OPENAI_API_KEY for unrelated tools.\n delete effectiveEnvRemnic.openaiApiKey;\n }\n return { ...fileRemnic, ...effectiveEnvRemnic };\n}\n\n// ── Helpers ─────────────────────────────────────────────────────────────────\n\nconst WRITABLE_BOOLEAN_CONFIG_KEYS = new Set([\n \"citationsAutoDetect\",\n \"citationsEnabled\",\n \"embeddingFallbackEnabled\",\n \"enrichmentAutoOnCreate\",\n \"enrichmentEnabled\",\n \"feedbackEnabled\",\n \"hostEmbeddingProviderEnabled\",\n \"localLlmDisableThinking\",\n \"localLlmEnabled\",\n \"localLlmFallback\",\n \"localLlmFastEnabled\",\n \"memoryExtensionsEnabled\",\n \"namespacesEnabled\",\n \"qmdEnabled\",\n \"queryExpansionEnabled\",\n \"recallPlannerEnabled\",\n \"recallPlannerLlmEnabled\",\n \"recallPlannerTelemetryEnabled\",\n \"rerankEnabled\",\n]);\n\nconst WRITABLE_STRING_CONFIG_KEYS = new Set([\n \"embeddingFallbackModel\",\n \"embeddingFallbackProvider\",\n \"fastGatewayAgentId\",\n \"gatewayAgentId\",\n \"localLlmFastModel\",\n \"localLlmModel\",\n \"localLlmUrl\",\n \"model\",\n \"modelSource\",\n \"openaiBaseUrl\",\n \"qmdEmbedModel\",\n \"qmdGenerateModel\",\n \"qmdRerankModel\",\n \"recallPlannerModel\",\n]);\n\nfunction isWritableConfigKey(key: string): boolean {\n return WRITABLE_BOOLEAN_CONFIG_KEYS.has(key) || WRITABLE_STRING_CONFIG_KEYS.has(key);\n}\n\nfunction hasEnv(name: string): boolean {\n const value = process.env[name];\n return typeof value === \"string\" && value.trim().length > 0;\n}\n\nfunction fileExists(candidate: string): boolean {\n try {\n return fs.existsSync(expandTildePath(candidate));\n } catch {\n return false;\n }\n}\n\nfunction canWriteConfigPath(configPath: string): boolean {\n try {\n if (fs.existsSync(configPath)) {\n fs.accessSync(configPath, fs.constants.W_OK);\n return true;\n }\n fs.accessSync(path.dirname(configPath), fs.constants.W_OK);\n return true;\n } catch {\n return false;\n }\n}\n\nfunction writeConfigFileAtomically(configPath: string, data: Record<string, unknown>): void {\n fs.mkdirSync(path.dirname(configPath), { recursive: true });\n const tmpPath = path.join(\n path.dirname(configPath),\n `.${path.basename(configPath)}.tmp-${process.pid}-${Date.now()}`,\n );\n let completed = false;\n try {\n fs.writeFileSync(tmpPath, `${JSON.stringify(data, null, 2)}\\n`, {\n encoding: \"utf8\",\n flag: \"wx\",\n mode: 0o600,\n });\n fs.renameSync(tmpPath, configPath);\n try {\n fs.chmodSync(configPath, 0o600);\n } catch {\n // Best effort for platforms/filesystems that do not support chmod.\n }\n completed = true;\n } finally {\n if (!completed) {\n try {\n fs.unlinkSync(tmpPath);\n } catch {\n // Best effort cleanup for failed writes.\n }\n }\n }\n}\n\nfunction readJsonRecordIfPresent(configPath: string): Record<string, unknown> {\n if (!fs.existsSync(configPath)) return {};\n const parsed = JSON.parse(fs.readFileSync(configPath, \"utf8\"));\n if (!isPlainRecord(parsed)) {\n throw new Error(`Invalid config file ${configPath}: top-level config must be a JSON object`);\n }\n return parsed;\n}\n\nfunction resolveEditableRemnicBlock(root: Record<string, unknown>): Record<string, unknown> {\n if (isPlainRecord(root.remnic)) return root.remnic;\n if (isPlainRecord(root.engram)) return root.engram;\n return root;\n}\n\nfunction normalizePatchValue(key: string, value: unknown): string | boolean | null {\n if (!isWritableConfigKey(key)) {\n throw new Error(`Unsupported admin config key: ${key}`);\n }\n if (value === null) return null;\n if (WRITABLE_BOOLEAN_CONFIG_KEYS.has(key)) {\n if (typeof value === \"boolean\") return value;\n throw new Error(`Invalid ${key}: expected boolean or null`);\n }\n if (typeof value !== \"string\") {\n throw new Error(`Invalid ${key}: expected string or null`);\n }\n const normalized = value.trim();\n if (key === \"modelSource\" && normalized !== \"plugin\" && normalized !== \"gateway\") {\n throw new Error(\"Invalid modelSource: expected plugin or gateway\");\n }\n if (\n key === \"embeddingFallbackProvider\" &&\n normalized !== \"auto\" &&\n normalized !== \"openai\" &&\n normalized !== \"local\"\n ) {\n throw new Error(\"Invalid embeddingFallbackProvider: expected auto, openai, or local\");\n }\n return normalized;\n}\n\nfunction publicConfigValues(config: PluginConfig, serverConfig: ParsedServerConfig): Record<string, string | number | boolean | null> {\n return {\n adminConsoleEnabled: serverConfig.adminConsoleEnabled,\n memoryDir: config.memoryDir,\n model: config.model,\n modelSource: config.modelSource,\n gatewayAgentId: config.gatewayAgentId || null,\n fastGatewayAgentId: config.fastGatewayAgentId || null,\n localLlmEnabled: config.localLlmEnabled,\n localLlmUrl: config.localLlmUrl || null,\n localLlmModel: config.localLlmModel || null,\n localLlmFastEnabled: config.localLlmFastEnabled,\n localLlmFastModel: config.localLlmFastModel || null,\n localLlmFallback: config.localLlmFallback,\n localLlmDisableThinking: config.localLlmDisableThinking,\n qmdEnabled: config.qmdEnabled,\n qmdEmbedModel: config.qmdEmbedModel || null,\n qmdRerankModel: config.qmdRerankModel || null,\n qmdGenerateModel: config.qmdGenerateModel || null,\n embeddingFallbackEnabled: config.embeddingFallbackEnabled,\n embeddingFallbackProvider: config.embeddingFallbackProvider,\n embeddingFallbackModel: config.embeddingFallbackModel || null,\n hostEmbeddingProviderEnabled: config.hostEmbeddingProviderEnabled,\n namespacesEnabled: config.namespacesEnabled,\n recallPlannerEnabled: config.recallPlannerEnabled,\n recallPlannerLlmEnabled: config.recallPlannerLlmEnabled,\n recallPlannerModel: config.recallPlannerModel || null,\n citationsEnabled: config.citationsEnabled,\n citationsAutoDetect: config.citationsAutoDetect,\n queryExpansionEnabled: config.queryExpansionEnabled,\n rerankEnabled: config.rerankEnabled,\n feedbackEnabled: config.feedbackEnabled,\n memoryExtensionsEnabled: config.memoryExtensionsEnabled,\n enrichmentEnabled: config.enrichmentEnabled,\n enrichmentAutoOnCreate: config.enrichmentAutoOnCreate,\n };\n}\n\nfunction configuredModels(config: PluginConfig): RemnicAdminModelOption[] {\n const models = new Map<string, RemnicAdminModelOption>();\n const add = (id: string | undefined, provider: string, label: string, enabled: boolean, isDefault = false, source = \"config\") => {\n const normalized = id?.trim();\n if (!normalized) return;\n const existing = models.get(`${provider}:${normalized}`);\n models.set(`${provider}:${normalized}`, {\n id: normalized,\n provider,\n label,\n detected: existing?.detected ?? true,\n enabled: existing?.enabled || enabled,\n default: existing?.default || isDefault,\n source,\n });\n };\n\n add(config.model, \"openai\", config.model, !isOpenaiApiKeyDisabled(config.openaiApiKey), config.modelSource === \"plugin\");\n add(config.gatewayAgentId, \"gateway\", config.gatewayAgentId, config.modelSource === \"gateway\", config.modelSource === \"gateway\");\n add(config.fastGatewayAgentId, \"gateway\", config.fastGatewayAgentId, config.modelSource === \"gateway\");\n add(config.localLlmModel, \"local\", config.localLlmModel, config.localLlmEnabled, config.localLlmEnabled);\n add(config.localLlmFastModel, \"local\", `${config.localLlmFastModel} (fast)`, config.localLlmFastEnabled);\n add(config.embeddingFallbackModel, config.embeddingFallbackProvider, `${config.embeddingFallbackModel} (embedding fallback)`, config.embeddingFallbackEnabled);\n add(config.qmdEmbedModel, \"qmd\", `${config.qmdEmbedModel} (embed)`, config.qmdEnabled);\n add(config.qmdRerankModel, \"qmd\", `${config.qmdRerankModel} (rerank)`, config.qmdEnabled);\n add(config.qmdGenerateModel, \"qmd\", `${config.qmdGenerateModel} (generate)`, config.qmdEnabled);\n add(config.recallPlannerModel, \"planner\", `${config.recallPlannerModel} (planner)`, config.recallPlannerLlmEnabled);\n\n return [...models.values()].sort((a, b) => `${a.provider}:${a.id}`.localeCompare(`${b.provider}:${b.id}`));\n}\n\nfunction isLikelyOllamaEndpoint(endpoint: string): boolean {\n return /(?:ollama|11434)/i.test(endpoint);\n}\n\nasync function detectOllamaModels(baseUrl: string | undefined): Promise<RemnicAdminModelOption[]> {\n const configuredEndpoint = baseUrl?.trim();\n const envEndpoint = process.env.OLLAMA_HOST?.trim();\n const endpoint =\n configuredEndpoint && isLikelyOllamaEndpoint(configuredEndpoint)\n ? configuredEndpoint\n : envEndpoint;\n if (!endpoint) return [];\n const controller = new AbortController();\n const timer = setTimeout(() => controller.abort(), 900);\n try {\n const url = new URL(\"/api/tags\", endpoint.endsWith(\"/\") ? endpoint : `${endpoint}/`);\n const response = await fetch(url, { signal: controller.signal });\n if (!response.ok) return [];\n const payload = await response.json() as { models?: Array<{ name?: unknown; model?: unknown }> };\n return (payload.models ?? [])\n .map((model) => typeof model.name === \"string\" ? model.name : typeof model.model === \"string\" ? model.model : \"\")\n .filter((name) => name.length > 0)\n .map((name) => ({\n id: name,\n label: name,\n provider: \"ollama\",\n detected: true,\n enabled: true,\n source: \"ollama\",\n endpoint,\n }));\n } catch {\n return [];\n } finally {\n clearTimeout(timer);\n }\n}\n\nfunction dashboardHarnesses(config: PluginConfig) {\n const openclawDetected = hasEnv(\"OPENCLAW_HOME\") || hasEnv(\"OPENCLAW_WORKSPACE\") || fileExists(\"~/.openclaw\");\n const codexDetected = hasEnv(\"CODEX_HOME\") || fileExists(\"~/.codex/auth.json\") || fileExists(\"~/.codex\");\n return [\n {\n id: \"remnic-http\",\n label: \"Remnic HTTP API\",\n detected: true,\n enabled: true,\n source: \"server\",\n detail: \"MCP and REST access server\",\n },\n {\n id: \"openclaw\",\n label: \"OpenClaw\",\n detected: openclawDetected,\n enabled: config.modelSource === \"gateway\" || config.hostEmbeddingProviderEnabled,\n source: openclawDetected ? \"host\" : \"not detected\",\n detail: \"Gateway and host adapters\",\n },\n {\n id: \"codex\",\n label: \"Codex\",\n detected: codexDetected,\n enabled: config.citationsEnabled || config.citationsAutoDetect,\n source: codexDetected ? \"host\" : \"not detected\",\n detail: \"Citation-aware adapter\",\n },\n {\n id: \"qmd\",\n label: \"QMD Search\",\n detected: Boolean(config.qmdPath) || config.qmdEnabled,\n enabled: config.qmdEnabled,\n source: config.qmdPath ? \"qmdPath\" : \"config\",\n detail: config.qmdSearchStrategy,\n },\n ];\n}\n\nfunction dashboardProviders(config: PluginConfig) {\n const localLlmUrl = config.localLlmUrl || \"\";\n const gatewayIds = [config.gatewayAgentId, config.fastGatewayAgentId].filter(Boolean).join(\" \");\n const openaiBaseUrl = process.env.OPENAI_BASE_URL || \"\";\n const sageRouterDetected =\n hasEnv(\"SAGE_ROUTER_URL\") ||\n hasEnv(\"SAGE_ROUTER_HOST\") ||\n /sage[-_ ]?router/i.test(`${gatewayIds} ${openaiBaseUrl}`);\n const ollamaDetected = hasEnv(\"OLLAMA_HOST\") || /(?:ollama|11434)/i.test(localLlmUrl);\n const localDetected = Boolean(localLlmUrl.trim());\n\n return [\n {\n id: \"openai\",\n label: \"OpenAI\",\n detected: !isOpenaiApiKeyDisabled(config.openaiApiKey),\n enabled: config.modelSource === \"plugin\" && !isOpenaiApiKeyDisabled(config.openaiApiKey),\n source: !isOpenaiApiKeyDisabled(config.openaiApiKey) ? \"config/env\" : \"disabled\",\n detail: config.model,\n },\n {\n id: \"sage-router\",\n label: \"Sage Router\",\n detected: sageRouterDetected,\n enabled: config.modelSource === \"gateway\" && sageRouterDetected,\n source: sageRouterDetected ? \"gateway/env\" : \"not detected\",\n detail: config.gatewayAgentId || config.fastGatewayAgentId || openaiBaseUrl || \"OpenAI-compatible provider router\",\n },\n {\n id: \"ollama\",\n label: \"Ollama\",\n detected: ollamaDetected,\n enabled: config.localLlmEnabled && ollamaDetected,\n source: ollamaDetected ? \"localLlmUrl/OLLAMA_HOST\" : \"not detected\",\n detail: \"Local or cloud-compatible Ollama endpoint\",\n },\n {\n id: \"local-openai-compatible\",\n label: \"Local OpenAI-compatible\",\n detected: localDetected && !ollamaDetected,\n enabled: config.localLlmEnabled && localDetected && !ollamaDetected,\n source: localDetected ? \"localLlmUrl\" : \"not detected\",\n detail: localLlmUrl || \"Local provider endpoint\",\n },\n ];\n}\n\nfunction dashboardFeatures(config: PluginConfig) {\n return [\n [\"localLlmEnabled\", \"Local LLM\", config.localLlmEnabled],\n [\"localLlmFastEnabled\", \"Fast Local Tier\", config.localLlmFastEnabled],\n [\"qmdEnabled\", \"QMD Search\", config.qmdEnabled],\n [\"embeddingFallbackEnabled\", \"Embedding Fallback\", config.embeddingFallbackEnabled],\n [\"hostEmbeddingProviderEnabled\", \"Host Embeddings\", config.hostEmbeddingProviderEnabled],\n [\"namespacesEnabled\", \"Namespaces\", config.namespacesEnabled],\n [\"recallPlannerEnabled\", \"Recall Planner\", config.recallPlannerEnabled],\n [\"recallPlannerLlmEnabled\", \"Planner LLM\", config.recallPlannerLlmEnabled],\n [\"citationsEnabled\", \"Citations\", config.citationsEnabled],\n [\"citationsAutoDetect\", \"Citation Auto-detect\", config.citationsAutoDetect],\n [\"queryExpansionEnabled\", \"Query Expansion\", config.queryExpansionEnabled],\n [\"rerankEnabled\", \"Rerank\", config.rerankEnabled],\n [\"feedbackEnabled\", \"Feedback\", config.feedbackEnabled],\n [\"memoryExtensionsEnabled\", \"Memory Extensions\", config.memoryExtensionsEnabled],\n [\"enrichmentEnabled\", \"Entity Enrichment\", config.enrichmentEnabled],\n ].map(([key, label, enabled]) => ({\n key: String(key),\n label: String(label),\n enabled: enabled === true,\n writable: WRITABLE_BOOLEAN_CONFIG_KEYS.has(String(key)),\n restartRequired: true,\n }));\n}\n\nfunction createAdminControls(\n configPath: string,\n config: PluginConfig,\n serverConfig: ParsedServerConfig,\n): RemnicAdminControls {\n let restartRequired = false;\n let displayConfig = config;\n const status = async (): Promise<RemnicAdminDashboardStatus> => {\n const models = configuredModels(displayConfig);\n const ollamaModels = await detectOllamaModels(displayConfig.localLlmUrl);\n const modelKeys = new Set(models.map((model) => `${model.provider}:${model.id}`));\n for (const model of ollamaModels) {\n if (!modelKeys.has(`${model.provider}:${model.id}`)) models.push(model);\n }\n return {\n config: {\n path: configPath,\n exists: fs.existsSync(configPath),\n writable: canWriteConfigPath(configPath),\n restartRequired,\n values: publicConfigValues(displayConfig, serverConfig),\n },\n harnesses: dashboardHarnesses(displayConfig),\n providers: dashboardProviders(displayConfig),\n models,\n features: dashboardFeatures(displayConfig),\n };\n };\n\n return {\n status,\n update: async (patch: RemnicAdminConfigPatch): Promise<RemnicAdminDashboardStatus> => {\n if (!isPlainRecord(patch)) {\n throw new Error(\"Admin config patch must be an object\");\n }\n const normalizedEntries = Object.entries(patch).map(([key, value]) => [key, normalizePatchValue(key, value)] as const);\n if (normalizedEntries.length === 0) return status();\n\n const raw = readJsonRecordIfPresent(configPath);\n const target = resolveEditableRemnicBlock(raw);\n for (const [key, value] of normalizedEntries) {\n if (value === null) {\n delete target[key];\n } else {\n target[key] = value;\n }\n }\n\n const candidateRemnic = isPlainRecord(raw.remnic)\n ? raw.remnic\n : isPlainRecord(raw.engram)\n ? raw.engram\n : raw;\n const nextDisplayConfig = parseConfig(candidateRemnic);\n\n writeConfigFileAtomically(configPath, raw);\n displayConfig = nextDisplayConfig;\n restartRequired = true;\n return status();\n },\n };\n}\n\n/**\n * Like `setTimeout` wrapped in a Promise, but respects an `AbortSignal`.\n * Resolves immediately (without throwing) when the signal fires so the\n * caller can check `signal.aborted` and exit cleanly.\n */\nfunction abortableDelay(ms: number, signal: AbortSignal): Promise<void> {\n if (signal.aborted) return Promise.resolve();\n return new Promise<void>((resolve) => {\n const timer = setTimeout(resolve, ms);\n const onAbort = () => {\n clearTimeout(timer);\n resolve();\n };\n signal.addEventListener(\"abort\", onAbort, { once: true });\n });\n}\n\nasync function cleanupFailedStartup(\n orchestrator: Orchestrator,\n httpServer: EngramAccessHttpServer,\n): Promise<void> {\n try {\n await httpServer.stop();\n } catch (err) {\n log.warn(`HTTP startup failure cleanup could not stop server: ${err}`);\n }\n\n try {\n await orchestrator.destroy();\n } catch (err) {\n log.warn(`HTTP startup failure cleanup could not destroy orchestrator: ${err}`);\n }\n}\n\n// ── Server startup ──────────────────────────────────────────────────────────\n\nexport interface ServerResult {\n config: PluginConfig;\n service: EngramAccessService;\n httpServer: EngramAccessHttpServer;\n host: string;\n port: number;\n /** Stop HTTP, cancel startup work, abort deferred init, and destroy the orchestrator. */\n stop: () => Promise<void>;\n /** Cancel any pending startup-sync retry timers. Called automatically on shutdown. */\n cancelStartupSync: () => void;\n /** Abort deferred orchestrator initialization (QMD sync, warmup, cache). */\n abortDeferredInit: () => void;\n}\n\nexport async function startServer(options?: {\n configPath?: string;\n host?: string;\n port?: number;\n authToken?: string;\n}): Promise<ServerResult> {\n initLogger();\n\n const resolvedConfigPath = resolveConfigPath(options?.configPath);\n const fileConfig = loadResolvedConfig(resolvedConfigPath);\n\n const env = envOverrides();\n const { remnic: envRemnic, ...envServer } = env;\n\n // Merge: file < env < cli flags\n const remnicConfig = mergeRemnicConfigForServer(fileConfig.remnic, envRemnic);\n const cliServerConfig: Partial<ServerConfig[\"server\"]> = {};\n if (options?.host !== undefined) cliServerConfig.host = options.host;\n if (options?.port !== undefined) cliServerConfig.port = parseServerPort(options.port, \"options.port\");\n if (options?.authToken !== undefined) cliServerConfig.authToken = options.authToken;\n\n const serverConfig = {\n ...fileConfig.server,\n ...envServer,\n ...cliServerConfig,\n };\n const portSource = cliServerConfig.port !== undefined\n ? \"options.port\"\n : envServer.port !== undefined\n ? \"REMNIC_PORT/ENGRAM_PORT\"\n : \"server.port\";\n const parsedServerConfig = parseServerConfig(serverConfig, { portSource });\n\n const config = parseConfig(remnicConfig);\n const orchestrator = new Orchestrator(config);\n await orchestrator.initialize();\n\n // Start the HTTP server immediately so health checks, MCP handshakes,\n // and liveness probes can connect while deferred init is still running.\n const service = new EngramAccessService(orchestrator);\n\n const authToken = parsedServerConfig.authToken ?? readCompatEnv(\"REMNIC_AUTH_TOKEN\", \"ENGRAM_AUTH_TOKEN\") ?? \"\";\n\n // Connector tokens are loaded dynamically per request via authTokensGetter\n // so that token generate/revoke takes effect without server restart\n if (!authToken && getAllValidTokens().length === 0) {\n log.warn(\"No auth token set — server will reject all requests. Set REMNIC_AUTH_TOKEN, server.authToken in config, or generate tokens with 'remnic token generate'.\");\n }\n\n const httpServer = new EngramAccessHttpServer({\n service,\n host: parsedServerConfig.host,\n port: parsedServerConfig.port,\n authToken: authToken || undefined,\n authTokensGetter: () => getAllValidTokensCached(),\n principal: parsedServerConfig.principal,\n maxBodyBytes: parsedServerConfig.maxBodyBytes,\n adminConsoleEnabled: parsedServerConfig.adminConsoleEnabled,\n adminConsolePublicDir: parsedServerConfig.adminConsolePublicDir\n ? path.resolve(expandTildePath(parsedServerConfig.adminConsolePublicDir))\n : undefined,\n adminConsolePrefillToken: parsedServerConfig.adminConsolePrefillToken,\n adminControls: parsedServerConfig.adminConsoleEnabled\n ? createAdminControls(resolvedConfigPath.path, config, parsedServerConfig)\n : undefined,\n citationsEnabled: config.citationsEnabled,\n citationsAutoDetect: config.citationsAutoDetect,\n emitLegacyTools: config.emitLegacyTools,\n });\n\n let host: string;\n let port: number;\n try {\n ({ host, port } = await httpServer.start());\n } catch (err) {\n await cleanupFailedStartup(orchestrator, httpServer);\n throw err;\n }\n\n // Fire-and-forget: wait for deferred init (QMD probe, collection setup,\n // warmup) then check QMD availability and retry if needed. This does NOT\n // block the server listener — connections are accepted immediately above.\n // An AbortController allows the shutdown handler to cancel pending retries.\n const startupSyncAbort = new AbortController();\n\n // Wrap httpServer.stop() so that existing callers also get full lifecycle\n // cleanup: retry timers, deferred init, HTTP listener, and orchestrator.\n const originalStop = httpServer.stop.bind(httpServer);\n let stopPromise: Promise<void> | undefined;\n const stop = async (): Promise<void> => {\n if (stopPromise) return stopPromise;\n stopPromise = (async () => {\n startupSyncAbort.abort();\n orchestrator.abortDeferredInit();\n try {\n await originalStop();\n } finally {\n await orchestrator.destroy();\n }\n })();\n return stopPromise;\n };\n httpServer.stop = stop;\n\n orchestrator.deferredReady.then(() => {\n if (startupSyncAbort.signal.aborted) {\n log.debug(\"QMD startup-sync: cancelled before deferred init completed\");\n return;\n }\n\n // Skip retries when search is explicitly disabled via config or when the\n // orchestrator already resolved to a noop backend (e.g. missing collection\n // detected during deferredInitialize). Both cases mean no sync should ever\n // run; scheduling retries would create misleading operational noise and\n // unnecessary background work on every server start.\n if (!config.qmdEnabled || orchestrator.qmd.debugStatus() === \"backend=noop\") {\n log.debug(\"QMD startup-sync: search disabled or noop backend, skipping retries\");\n return;\n }\n\n // Retry when either: (a) QMD is not available yet (cold-start race), or\n // (b) QMD is available but the deferred init sync step failed silently\n // (e.g., update errors swallowed by backend, throttle skip, transient\n // network failure). Without (b), the daemon permanently serves stale\n // recall after a failed sync despite healthy QMD probe.\n const needsRetry = !orchestrator.qmd.isAvailable() || !orchestrator.deferredSyncSucceeded;\n if (!needsRetry) {\n log.debug(\"QMD startup-sync: deferred init completed successfully, no retries needed\");\n return;\n }\n\n const RETRY_DELAYS_MS = [5_000, 15_000, 30_000, 60_000, 120_000];\n if (startupSyncAbort.signal.aborted) {\n log.debug(\"QMD startup-sync retry: cancelled before retry task started\");\n return;\n }\n (async () => {\n for (const delay of RETRY_DELAYS_MS) {\n await abortableDelay(delay, startupSyncAbort.signal);\n\n if (startupSyncAbort.signal.aborted) {\n log.debug(\"QMD startup-sync retry: cancelled by shutdown\");\n return;\n }\n\n const synced = await orchestrator.startupSearchSync(startupSyncAbort.signal);\n if (!synced) {\n if (orchestrator.qmd.debugStatus() === \"backend=noop\") {\n log.debug(\"QMD startup-sync retry: search intentionally disabled; stopping retries\");\n return;\n }\n log.debug(`QMD startup-sync retry: not available yet (next retry in ${RETRY_DELAYS_MS[RETRY_DELAYS_MS.indexOf(delay) + 1] ?? \"n/a\"}ms)`);\n continue;\n }\n\n return; // sync succeeded, stop retrying\n }\n\n log.warn(\"QMD startup-sync retry: exhausted all retries; search index may be stale\");\n })().catch((err: unknown) => {\n log.warn(`QMD startup-sync retry: unexpected error: ${err}`);\n });\n }).catch((err: unknown) => {\n log.warn(`Deferred init error: ${err}`);\n });\n\n return { config, service, httpServer, host, port, stop, cancelStartupSync: () => startupSyncAbort.abort(), abortDeferredInit: () => orchestrator.abortDeferredInit() };\n}\n\n// ── CLI entry point ──────────────────────────────────────────────────────────\n\nconst BOOLEAN_CLI_OPTIONS = new Set([\"help\"]);\nconst VALUE_CLI_OPTIONS = new Set([\"config\", \"host\", \"port\", \"auth-token\"]);\n\nfunction parseCliArgs(argv: string[]): Record<string, string | undefined> {\n const args: Record<string, string | undefined> = {};\n for (let i = 0; i < argv.length; i++) {\n const token = argv[i];\n if (token === \"-h\") {\n args.help = \"true\";\n continue;\n }\n\n if (token.startsWith(\"--\")) {\n const [key, inlineValue] = token.slice(2).split(/=(.*)/s, 2);\n if (!key) {\n throw new Error(`Invalid option ${token}`);\n }\n\n if (BOOLEAN_CLI_OPTIONS.has(key)) {\n if (inlineValue !== undefined) {\n throw new Error(`Option --${key} does not accept a value`);\n }\n args[key] = \"true\";\n continue;\n }\n\n if (!VALUE_CLI_OPTIONS.has(key)) {\n throw new Error(`Unknown option --${key}`);\n }\n\n const value = inlineValue ?? argv[i + 1];\n if (\n value === undefined ||\n (inlineValue === undefined && value.startsWith(\"--\")) ||\n value.trim() === \"\"\n ) {\n throw new Error(`Missing value for --${key}`);\n }\n\n args[key] = value;\n if (inlineValue === undefined) i++;\n }\n }\n return args;\n}\n\nexport async function cliMain(argv: string[] = process.argv.slice(2)): Promise<void> {\n const args = parseCliArgs(argv);\n\n if (args.help) {\n console.log(`\nremnic-server — Standalone Remnic memory server\n\nUsage:\n remnic-server [options]\n\nOptions:\n --config <path> Path to config file (default: remnic.config.json)\n --host <addr> Bind address (default: 127.0.0.1)\n --port <number> Port number (default: 4318)\n --auth-token <tok> Bearer token for auth (or set REMNIC_AUTH_TOKEN)\n --help Show this help\n\nEnvironment:\n REMNIC_CONFIG_PATH Config file path (ENGRAM_CONFIG_PATH also supported)\n REMNIC_PORT Server port (ENGRAM_PORT also supported)\n REMNIC_HOST Bind address (ENGRAM_HOST also supported)\n REMNIC_AUTH_TOKEN Auth bearer token (ENGRAM_AUTH_TOKEN also supported)\n REMNIC_ADMIN_CONSOLE_PREFILL_TOKEN\n Prefill admin UI with REMNIC_AUTH_TOKEN when true\n REMNIC_MEMORY_DIR Override memory directory (ENGRAM_MEMORY_DIR also supported)\n OPENAI_API_KEY OpenAI API key for extraction; ignored when config sets openaiApiKey=false\n`);\n process.exit(0);\n }\n\n const result = await startServer({\n configPath: args.config,\n host: args.host,\n port: args.port === undefined ? undefined : parseServerPort(args.port, \"--port\"),\n authToken: args[\"auth-token\"],\n });\n\n console.log(`Remnic server listening on http://${result.host}:${result.port}`);\n\n // Graceful shutdown\n const shutdown = async (signal: string) => {\n console.log(`\\nReceived ${signal}, shutting down...`);\n await result.stop();\n process.exit(0);\n };\n\n process.on(\"SIGINT\", () => shutdown(\"SIGINT\"));\n process.on(\"SIGTERM\", () => shutdown(\"SIGTERM\"));\n}\n\n// Auto-run when executed directly\n// Matches direct execution of `node .../remnic-server/dist/index.js` or\n// `node .../remnic-server/src/index.ts`. Package command names are handled by\n// the bin wrappers in ../bin so importing this module cannot start twice.\nif (\n process.argv[1] &&\n /(?:remnic-server|engram-server)[\\\\/](?:dist|src)[\\\\/]index\\.[jt]s$/.test(process.argv[1])\n) {\n cliMain().catch((err) => {\n process.stderr.write(`Fatal: ${err instanceof Error ? err.message : String(err)}\\n`);\n process.exit(1);\n });\n}\n"],"mappings":"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","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@remnic/server",
|
|
3
|
-
"version": "9.3.
|
|
3
|
+
"version": "9.3.639",
|
|
4
4
|
"description": "Standalone Remnic memory server — HTTP + MCP without OpenClaw",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -24,12 +24,12 @@
|
|
|
24
24
|
"provenance": true
|
|
25
25
|
},
|
|
26
26
|
"dependencies": {
|
|
27
|
-
"@remnic/core": "^9.3.
|
|
27
|
+
"@remnic/core": "^9.3.639"
|
|
28
28
|
},
|
|
29
29
|
"devDependencies": {
|
|
30
30
|
"tsup": "^8.5.1",
|
|
31
31
|
"typescript": "^5.9.3",
|
|
32
|
-
"@remnic/core": "^9.3.
|
|
32
|
+
"@remnic/core": "^9.3.639"
|
|
33
33
|
},
|
|
34
34
|
"license": "MIT",
|
|
35
35
|
"repository": {
|