@remnic/server 1.0.4 → 9.3.515
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 +18 -5
- package/bin/engram-server.js +4 -0
- package/bin/remnic-server.js +4 -0
- package/bin/server-bin.js +41 -0
- package/dist/index.d.ts +64 -0
- package/dist/index.js +252 -33
- package/dist/index.js.map +1 -1
- package/package.json +12 -6
- package/dist/bin/chunk-MKNXEH3P.js +0 -152
- package/dist/bin/chunk-MKNXEH3P.js.map +0 -1
- package/dist/bin/engram-server.js +0 -13
- package/dist/bin/engram-server.js.map +0 -1
- package/dist/bin/remnic-server.js +0 -13
- package/dist/bin/remnic-server.js.map +0 -1
package/README.md
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
# @remnic/server
|
|
2
2
|
|
|
3
|
-
Standalone Remnic memory server -- HTTP and MCP interfaces without requiring OpenClaw.
|
|
3
|
+
Standalone Remnic memory and context server -- HTTP and MCP interfaces without requiring OpenClaw.
|
|
4
4
|
|
|
5
|
-
Part of [Remnic](https://github.com/joshuaswarren/remnic),
|
|
5
|
+
Part of [Remnic](https://github.com/joshuaswarren/remnic), open-source memory and context for user-aware agents.
|
|
6
6
|
|
|
7
7
|
## Install
|
|
8
8
|
|
|
@@ -23,15 +23,28 @@ Both interfaces connect to the same [`@remnic/core`](https://www.npmjs.com/packa
|
|
|
23
23
|
|
|
24
24
|
## Usage
|
|
25
25
|
|
|
26
|
+
Run the standalone server:
|
|
27
|
+
|
|
28
|
+
```bash
|
|
29
|
+
npx --package @remnic/server remnic-server --help
|
|
30
|
+
npx --package @remnic/server remnic-server --port 4318
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
The package also ships the legacy `engram-server` binary for compatibility.
|
|
34
|
+
The bin wrappers are source-controlled so package managers can link them during
|
|
35
|
+
workspace installs; release builds verify that both targets have Node shebangs
|
|
36
|
+
and can start their help command before publish.
|
|
37
|
+
|
|
26
38
|
```typescript
|
|
27
|
-
import {
|
|
39
|
+
import { startServer } from "@remnic/server";
|
|
28
40
|
|
|
29
|
-
const server =
|
|
41
|
+
const server = await startServer({
|
|
30
42
|
port: 3141,
|
|
31
43
|
authToken: process.env.REMNIC_AUTH_TOKEN,
|
|
32
44
|
});
|
|
33
45
|
|
|
34
|
-
|
|
46
|
+
console.log(`Remnic server listening on http://${server.host}:${server.port}`);
|
|
47
|
+
await server.stop();
|
|
35
48
|
```
|
|
36
49
|
|
|
37
50
|
## License
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
export function shouldPrintHelpWithoutCli(argv) {
|
|
2
|
+
return argv.length === 1 && (argv[0] === "--help" || argv[0] === "-h");
|
|
3
|
+
}
|
|
4
|
+
|
|
5
|
+
export async function runServerBin(commandName, options = {}) {
|
|
6
|
+
const argv = options.argv ?? process.argv.slice(2);
|
|
7
|
+
const help = `
|
|
8
|
+
${commandName} - Standalone Remnic memory server
|
|
9
|
+
|
|
10
|
+
Usage:
|
|
11
|
+
${commandName} [options]
|
|
12
|
+
|
|
13
|
+
Options:
|
|
14
|
+
--config <path> Path to config file (default: remnic.config.json)
|
|
15
|
+
--host <addr> Bind address (default: 127.0.0.1)
|
|
16
|
+
--port <number> Port number (default: 4318)
|
|
17
|
+
--auth-token <tok> Bearer token for auth (or set REMNIC_AUTH_TOKEN)
|
|
18
|
+
--help Show this help
|
|
19
|
+
|
|
20
|
+
Environment:
|
|
21
|
+
REMNIC_CONFIG_PATH Config file path (ENGRAM_CONFIG_PATH also supported)
|
|
22
|
+
REMNIC_PORT Server port (ENGRAM_PORT also supported)
|
|
23
|
+
REMNIC_HOST Bind address (ENGRAM_HOST also supported)
|
|
24
|
+
REMNIC_AUTH_TOKEN Auth bearer token (ENGRAM_AUTH_TOKEN also supported)
|
|
25
|
+
REMNIC_MEMORY_DIR Override memory directory (ENGRAM_MEMORY_DIR also supported)
|
|
26
|
+
OPENAI_API_KEY OpenAI API key for extraction; ignored when config sets openaiApiKey=false
|
|
27
|
+
`;
|
|
28
|
+
|
|
29
|
+
if (shouldPrintHelpWithoutCli(argv)) {
|
|
30
|
+
(options.stdout ?? console.log)(help);
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const loadCliMain = options.loadCliMain ?? (() => import("../dist/index.js"));
|
|
35
|
+
const { cliMain } = await loadCliMain();
|
|
36
|
+
|
|
37
|
+
await cliMain(argv).catch((err) => {
|
|
38
|
+
(options.stderr ?? process.stderr.write.bind(process.stderr))(`Fatal: ${err instanceof Error ? err.message : String(err)}\n`);
|
|
39
|
+
(options.exit ?? process.exit)(1);
|
|
40
|
+
});
|
|
41
|
+
}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { PluginConfig, EngramAccessService, EngramAccessHttpServer } from '@remnic/core';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* @remnic/server
|
|
5
|
+
*
|
|
6
|
+
* Standalone Remnic memory server.
|
|
7
|
+
*
|
|
8
|
+
* Loads config from `remnic.config.json` (or env vars), creates an Orchestrator,
|
|
9
|
+
* and starts the HTTP access server with MCP endpoint — no OpenClaw required.
|
|
10
|
+
*
|
|
11
|
+
* Usage:
|
|
12
|
+
* npx --package @remnic/server remnic-server
|
|
13
|
+
* npx --package @remnic/server remnic-server --config ./my-remnic.json
|
|
14
|
+
* npx --package @remnic/server remnic-server --port 4320
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
interface ServerConfig {
|
|
18
|
+
remnic: Record<string, unknown>;
|
|
19
|
+
server: {
|
|
20
|
+
host?: string;
|
|
21
|
+
port?: unknown;
|
|
22
|
+
authToken?: string;
|
|
23
|
+
principal?: string;
|
|
24
|
+
maxBodyBytes?: number;
|
|
25
|
+
adminConsoleEnabled?: boolean;
|
|
26
|
+
adminConsolePublicDir?: string;
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
interface ParsedServerConfig {
|
|
30
|
+
host: string;
|
|
31
|
+
port: number;
|
|
32
|
+
authToken?: string;
|
|
33
|
+
principal?: string;
|
|
34
|
+
maxBodyBytes?: number;
|
|
35
|
+
adminConsoleEnabled: boolean;
|
|
36
|
+
adminConsolePublicDir?: string;
|
|
37
|
+
}
|
|
38
|
+
declare function parseServerConfig(raw: Partial<ServerConfig["server"]>, options?: {
|
|
39
|
+
portSource?: string;
|
|
40
|
+
}): ParsedServerConfig;
|
|
41
|
+
declare function loadConfigFile(configPath: string): ServerConfig;
|
|
42
|
+
declare function mergeRemnicConfigForServer(fileRemnic: Record<string, unknown>, envRemnic: Record<string, unknown> | undefined): Record<string, unknown>;
|
|
43
|
+
interface ServerResult {
|
|
44
|
+
config: PluginConfig;
|
|
45
|
+
service: EngramAccessService;
|
|
46
|
+
httpServer: EngramAccessHttpServer;
|
|
47
|
+
host: string;
|
|
48
|
+
port: number;
|
|
49
|
+
/** Stop HTTP, cancel startup work, abort deferred init, and destroy the orchestrator. */
|
|
50
|
+
stop: () => Promise<void>;
|
|
51
|
+
/** Cancel any pending startup-sync retry timers. Called automatically on shutdown. */
|
|
52
|
+
cancelStartupSync: () => void;
|
|
53
|
+
/** Abort deferred orchestrator initialization (QMD sync, warmup, cache). */
|
|
54
|
+
abortDeferredInit: () => void;
|
|
55
|
+
}
|
|
56
|
+
declare function startServer(options?: {
|
|
57
|
+
configPath?: string;
|
|
58
|
+
host?: string;
|
|
59
|
+
port?: number;
|
|
60
|
+
authToken?: string;
|
|
61
|
+
}): Promise<ServerResult>;
|
|
62
|
+
declare function cliMain(argv?: string[]): Promise<void>;
|
|
63
|
+
|
|
64
|
+
export { type ParsedServerConfig, type ServerConfig, type ServerResult, cliMain, loadConfigFile, mergeRemnicConfigForServer, parseServerConfig, startServer };
|
package/dist/index.js
CHANGED
|
@@ -3,14 +3,72 @@
|
|
|
3
3
|
// src/index.ts
|
|
4
4
|
import fs from "fs";
|
|
5
5
|
import path from "path";
|
|
6
|
-
import { parseConfig, Orchestrator, EngramAccessService, EngramAccessHttpServer, initLogger, log, getAllValidTokens, getAllValidTokensCached } from "@remnic/core";
|
|
6
|
+
import { parseConfig, isOpenaiApiKeyDisabled, Orchestrator, EngramAccessService, EngramAccessHttpServer, initLogger, log, getAllValidTokens, getAllValidTokensCached, expandTildePath } from "@remnic/core";
|
|
7
7
|
function readCompatEnv(primary, legacy) {
|
|
8
8
|
return process.env[primary] ?? process.env[legacy];
|
|
9
9
|
}
|
|
10
|
+
function parseServerPort(value, source) {
|
|
11
|
+
const port = typeof value === "string" ? Number(value.trim()) : value;
|
|
12
|
+
if (typeof port !== "number" || !Number.isInteger(port) || port < 1 || port > 65535) {
|
|
13
|
+
throw new Error(`Invalid ${source}: expected an integer port from 1 to 65535`);
|
|
14
|
+
}
|
|
15
|
+
return port;
|
|
16
|
+
}
|
|
17
|
+
function parseOptionalString(value, source) {
|
|
18
|
+
if (value === void 0) return void 0;
|
|
19
|
+
if (typeof value !== "string") {
|
|
20
|
+
throw new Error(`Invalid ${source}: expected a string`);
|
|
21
|
+
}
|
|
22
|
+
return value;
|
|
23
|
+
}
|
|
24
|
+
function parseOptionalNonEmptyString(value, source) {
|
|
25
|
+
const parsed = parseOptionalString(value, source);
|
|
26
|
+
if (parsed === void 0) return void 0;
|
|
27
|
+
if (parsed.trim() === "") {
|
|
28
|
+
throw new Error(`Invalid ${source}: expected a non-empty string`);
|
|
29
|
+
}
|
|
30
|
+
return parsed;
|
|
31
|
+
}
|
|
32
|
+
function parseOptionalPositiveInteger(value, source) {
|
|
33
|
+
if (value === void 0) return void 0;
|
|
34
|
+
const parsed = typeof value === "string" ? Number(value.trim()) : value;
|
|
35
|
+
if (typeof parsed !== "number" || !Number.isInteger(parsed) || parsed < 1) {
|
|
36
|
+
throw new Error(`Invalid ${source}: expected a positive integer`);
|
|
37
|
+
}
|
|
38
|
+
return parsed;
|
|
39
|
+
}
|
|
40
|
+
function parseOptionalBoolean(value, source) {
|
|
41
|
+
if (value === void 0) return void 0;
|
|
42
|
+
if (typeof value === "boolean") return value;
|
|
43
|
+
if (typeof value === "string") {
|
|
44
|
+
const normalized = value.trim().toLowerCase();
|
|
45
|
+
if (["true", "1", "yes", "on"].includes(normalized)) return true;
|
|
46
|
+
if (["false", "0", "no", "off"].includes(normalized)) return false;
|
|
47
|
+
}
|
|
48
|
+
throw new Error(`Invalid ${source}: expected a boolean`);
|
|
49
|
+
}
|
|
50
|
+
function parseServerConfig(raw, options) {
|
|
51
|
+
return {
|
|
52
|
+
host: parseOptionalNonEmptyString(raw.host, "server.host") ?? "127.0.0.1",
|
|
53
|
+
port: raw.port === void 0 ? 4318 : parseServerPort(raw.port, options?.portSource ?? "server.port"),
|
|
54
|
+
authToken: parseOptionalString(raw.authToken, "server.authToken"),
|
|
55
|
+
principal: parseOptionalString(raw.principal, "server.principal"),
|
|
56
|
+
maxBodyBytes: parseOptionalPositiveInteger(raw.maxBodyBytes, "server.maxBodyBytes"),
|
|
57
|
+
adminConsoleEnabled: parseOptionalBoolean(raw.adminConsoleEnabled, "server.adminConsoleEnabled") ?? false,
|
|
58
|
+
adminConsolePublicDir: parseOptionalString(raw.adminConsolePublicDir, "server.adminConsolePublicDir")
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
function resolveUserPath(value) {
|
|
62
|
+
return path.resolve(expandTildePath(value));
|
|
63
|
+
}
|
|
10
64
|
function resolveConfigPath(cliPath) {
|
|
11
|
-
if (cliPath)
|
|
65
|
+
if (cliPath) {
|
|
66
|
+
return { path: resolveUserPath(cliPath), explicit: true, source: "--config" };
|
|
67
|
+
}
|
|
12
68
|
const envPath = readCompatEnv("REMNIC_CONFIG_PATH", "ENGRAM_CONFIG_PATH");
|
|
13
|
-
if (envPath)
|
|
69
|
+
if (envPath) {
|
|
70
|
+
return { path: resolveUserPath(envPath), explicit: true, source: "REMNIC_CONFIG_PATH/ENGRAM_CONFIG_PATH" };
|
|
71
|
+
}
|
|
14
72
|
const homeDir = process.env.HOME ?? "~";
|
|
15
73
|
const candidates = [
|
|
16
74
|
path.join(process.cwd(), "remnic.config.json"),
|
|
@@ -19,24 +77,59 @@ function resolveConfigPath(cliPath) {
|
|
|
19
77
|
path.join(homeDir, ".config", "engram", "config.json")
|
|
20
78
|
];
|
|
21
79
|
for (const candidate of candidates) {
|
|
22
|
-
if (fs.existsSync(candidate)
|
|
80
|
+
if (fs.existsSync(candidate) && fs.statSync(candidate).isFile()) {
|
|
81
|
+
return { path: candidate, explicit: false, source: "auto-discovery" };
|
|
82
|
+
}
|
|
23
83
|
}
|
|
24
|
-
return path.join(homeDir, ".config", "remnic", "config.json");
|
|
84
|
+
return { path: path.join(homeDir, ".config", "remnic", "config.json"), explicit: false, source: "auto-discovery" };
|
|
85
|
+
}
|
|
86
|
+
function isPlainRecord(value) {
|
|
87
|
+
return !!value && typeof value === "object" && !Array.isArray(value);
|
|
88
|
+
}
|
|
89
|
+
function requirePlainConfigBlock(raw, key, configPath) {
|
|
90
|
+
const value = raw[key];
|
|
91
|
+
if (value === void 0) return void 0;
|
|
92
|
+
if (!isPlainRecord(value)) {
|
|
93
|
+
throw new Error(`Invalid config file ${configPath}: ${key} must be a JSON object`);
|
|
94
|
+
}
|
|
95
|
+
return value;
|
|
25
96
|
}
|
|
26
97
|
function loadConfigFile(configPath) {
|
|
27
98
|
const raw = JSON.parse(fs.readFileSync(configPath, "utf8"));
|
|
99
|
+
if (!isPlainRecord(raw)) {
|
|
100
|
+
throw new Error(`Invalid config file ${configPath}: top-level config must be a JSON object`);
|
|
101
|
+
}
|
|
102
|
+
const remnic = requirePlainConfigBlock(raw, "remnic", configPath);
|
|
103
|
+
const engram = requirePlainConfigBlock(raw, "engram", configPath);
|
|
104
|
+
const server = requirePlainConfigBlock(raw, "server", configPath);
|
|
28
105
|
return {
|
|
29
|
-
remnic:
|
|
30
|
-
server:
|
|
106
|
+
remnic: remnic ?? engram ?? raw,
|
|
107
|
+
server: server ?? {}
|
|
31
108
|
};
|
|
32
109
|
}
|
|
110
|
+
function loadResolvedConfig(resolved) {
|
|
111
|
+
if (!fs.existsSync(resolved.path)) {
|
|
112
|
+
if (resolved.explicit) {
|
|
113
|
+
throw new Error(`Config file from ${resolved.source} not found: ${resolved.path}`);
|
|
114
|
+
}
|
|
115
|
+
return { remnic: {}, server: {} };
|
|
116
|
+
}
|
|
117
|
+
const stat = fs.statSync(resolved.path);
|
|
118
|
+
if (!stat.isFile()) {
|
|
119
|
+
if (!resolved.explicit) {
|
|
120
|
+
return { remnic: {}, server: {} };
|
|
121
|
+
}
|
|
122
|
+
throw new Error(`Config file from ${resolved.source} is not a regular file: ${resolved.path}`);
|
|
123
|
+
}
|
|
124
|
+
return loadConfigFile(resolved.path);
|
|
125
|
+
}
|
|
33
126
|
function envOverrides() {
|
|
34
127
|
const overrides = {};
|
|
35
128
|
const remnic = {};
|
|
36
129
|
const port = readCompatEnv("REMNIC_PORT", "ENGRAM_PORT");
|
|
37
130
|
const host = readCompatEnv("REMNIC_HOST", "ENGRAM_HOST");
|
|
38
131
|
const authToken = readCompatEnv("REMNIC_AUTH_TOKEN", "ENGRAM_AUTH_TOKEN");
|
|
39
|
-
if (port) overrides.port =
|
|
132
|
+
if (port) overrides.port = port;
|
|
40
133
|
if (host) overrides.host = host;
|
|
41
134
|
if (authToken) overrides.authToken = authToken;
|
|
42
135
|
if (process.env.OPENAI_API_KEY) remnic.openaiApiKey = process.env.OPENAI_API_KEY;
|
|
@@ -44,54 +137,177 @@ function envOverrides() {
|
|
|
44
137
|
if (memoryDir) remnic.memoryDir = memoryDir;
|
|
45
138
|
return { ...overrides, ...Object.keys(remnic).length > 0 ? { remnic } : {} };
|
|
46
139
|
}
|
|
140
|
+
function mergeRemnicConfigForServer(fileRemnic, envRemnic) {
|
|
141
|
+
const effectiveEnvRemnic = { ...envRemnic ?? {} };
|
|
142
|
+
if (isOpenaiApiKeyDisabled(fileRemnic.openaiApiKey)) {
|
|
143
|
+
delete effectiveEnvRemnic.openaiApiKey;
|
|
144
|
+
}
|
|
145
|
+
return { ...fileRemnic, ...effectiveEnvRemnic };
|
|
146
|
+
}
|
|
147
|
+
function abortableDelay(ms, signal) {
|
|
148
|
+
if (signal.aborted) return Promise.resolve();
|
|
149
|
+
return new Promise((resolve) => {
|
|
150
|
+
const timer = setTimeout(resolve, ms);
|
|
151
|
+
const onAbort = () => {
|
|
152
|
+
clearTimeout(timer);
|
|
153
|
+
resolve();
|
|
154
|
+
};
|
|
155
|
+
signal.addEventListener("abort", onAbort, { once: true });
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
async function cleanupFailedStartup(orchestrator, httpServer) {
|
|
159
|
+
try {
|
|
160
|
+
await httpServer.stop();
|
|
161
|
+
} catch (err) {
|
|
162
|
+
log.warn(`HTTP startup failure cleanup could not stop server: ${err}`);
|
|
163
|
+
}
|
|
164
|
+
try {
|
|
165
|
+
await orchestrator.destroy();
|
|
166
|
+
} catch (err) {
|
|
167
|
+
log.warn(`HTTP startup failure cleanup could not destroy orchestrator: ${err}`);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
47
170
|
async function startServer(options) {
|
|
48
171
|
initLogger();
|
|
49
|
-
const
|
|
50
|
-
const fileConfig =
|
|
172
|
+
const resolvedConfigPath = resolveConfigPath(options?.configPath);
|
|
173
|
+
const fileConfig = loadResolvedConfig(resolvedConfigPath);
|
|
51
174
|
const env = envOverrides();
|
|
52
|
-
const
|
|
175
|
+
const { remnic: envRemnic, ...envServer } = env;
|
|
176
|
+
const remnicConfig = mergeRemnicConfigForServer(fileConfig.remnic, envRemnic);
|
|
177
|
+
const cliServerConfig = {};
|
|
178
|
+
if (options?.host !== void 0) cliServerConfig.host = options.host;
|
|
179
|
+
if (options?.port !== void 0) cliServerConfig.port = parseServerPort(options.port, "options.port");
|
|
180
|
+
if (options?.authToken !== void 0) cliServerConfig.authToken = options.authToken;
|
|
53
181
|
const serverConfig = {
|
|
54
182
|
...fileConfig.server,
|
|
55
|
-
...
|
|
56
|
-
...
|
|
57
|
-
...options?.port ? { port: options.port } : {},
|
|
58
|
-
...options?.authToken ? { authToken: options.authToken } : {}
|
|
183
|
+
...envServer,
|
|
184
|
+
...cliServerConfig
|
|
59
185
|
};
|
|
186
|
+
const portSource = cliServerConfig.port !== void 0 ? "options.port" : envServer.port !== void 0 ? "REMNIC_PORT/ENGRAM_PORT" : "server.port";
|
|
187
|
+
const parsedServerConfig = parseServerConfig(serverConfig, { portSource });
|
|
60
188
|
const config = parseConfig(remnicConfig);
|
|
61
189
|
const orchestrator = new Orchestrator(config);
|
|
190
|
+
await orchestrator.initialize();
|
|
62
191
|
const service = new EngramAccessService(orchestrator);
|
|
63
|
-
const authToken =
|
|
192
|
+
const authToken = parsedServerConfig.authToken ?? readCompatEnv("REMNIC_AUTH_TOKEN", "ENGRAM_AUTH_TOKEN") ?? "";
|
|
64
193
|
if (!authToken && getAllValidTokens().length === 0) {
|
|
65
194
|
log.warn("No auth token set \u2014 server will reject all requests. Set REMNIC_AUTH_TOKEN, server.authToken in config, or generate tokens with 'remnic token generate'.");
|
|
66
195
|
}
|
|
67
196
|
const httpServer = new EngramAccessHttpServer({
|
|
68
197
|
service,
|
|
69
|
-
host:
|
|
70
|
-
port:
|
|
198
|
+
host: parsedServerConfig.host,
|
|
199
|
+
port: parsedServerConfig.port,
|
|
71
200
|
authToken: authToken || void 0,
|
|
72
201
|
authTokensGetter: () => getAllValidTokensCached(),
|
|
73
|
-
principal:
|
|
74
|
-
maxBodyBytes:
|
|
75
|
-
adminConsoleEnabled:
|
|
202
|
+
principal: parsedServerConfig.principal,
|
|
203
|
+
maxBodyBytes: parsedServerConfig.maxBodyBytes,
|
|
204
|
+
adminConsoleEnabled: parsedServerConfig.adminConsoleEnabled,
|
|
205
|
+
adminConsolePublicDir: parsedServerConfig.adminConsolePublicDir ? path.resolve(expandTildePath(parsedServerConfig.adminConsolePublicDir)) : void 0,
|
|
76
206
|
citationsEnabled: config.citationsEnabled,
|
|
77
207
|
citationsAutoDetect: config.citationsAutoDetect
|
|
78
208
|
});
|
|
79
|
-
|
|
80
|
-
|
|
209
|
+
let host;
|
|
210
|
+
let port;
|
|
211
|
+
try {
|
|
212
|
+
({ host, port } = await httpServer.start());
|
|
213
|
+
} catch (err) {
|
|
214
|
+
await cleanupFailedStartup(orchestrator, httpServer);
|
|
215
|
+
throw err;
|
|
216
|
+
}
|
|
217
|
+
const startupSyncAbort = new AbortController();
|
|
218
|
+
const originalStop = httpServer.stop.bind(httpServer);
|
|
219
|
+
let stopPromise;
|
|
220
|
+
const stop = async () => {
|
|
221
|
+
if (stopPromise) return stopPromise;
|
|
222
|
+
stopPromise = (async () => {
|
|
223
|
+
startupSyncAbort.abort();
|
|
224
|
+
orchestrator.abortDeferredInit();
|
|
225
|
+
try {
|
|
226
|
+
await originalStop();
|
|
227
|
+
} finally {
|
|
228
|
+
await orchestrator.destroy();
|
|
229
|
+
}
|
|
230
|
+
})();
|
|
231
|
+
return stopPromise;
|
|
232
|
+
};
|
|
233
|
+
httpServer.stop = stop;
|
|
234
|
+
orchestrator.deferredReady.then(() => {
|
|
235
|
+
if (startupSyncAbort.signal.aborted) {
|
|
236
|
+
log.debug("QMD startup-sync: cancelled before deferred init completed");
|
|
237
|
+
return;
|
|
238
|
+
}
|
|
239
|
+
if (!config.qmdEnabled || orchestrator.qmd.debugStatus() === "backend=noop") {
|
|
240
|
+
log.debug("QMD startup-sync: search disabled or noop backend, skipping retries");
|
|
241
|
+
return;
|
|
242
|
+
}
|
|
243
|
+
const needsRetry = !orchestrator.qmd.isAvailable() || !orchestrator.deferredSyncSucceeded;
|
|
244
|
+
if (!needsRetry) {
|
|
245
|
+
log.debug("QMD startup-sync: deferred init completed successfully, no retries needed");
|
|
246
|
+
return;
|
|
247
|
+
}
|
|
248
|
+
const RETRY_DELAYS_MS = [5e3, 15e3, 3e4, 6e4, 12e4];
|
|
249
|
+
if (startupSyncAbort.signal.aborted) {
|
|
250
|
+
log.debug("QMD startup-sync retry: cancelled before retry task started");
|
|
251
|
+
return;
|
|
252
|
+
}
|
|
253
|
+
(async () => {
|
|
254
|
+
for (const delay of RETRY_DELAYS_MS) {
|
|
255
|
+
await abortableDelay(delay, startupSyncAbort.signal);
|
|
256
|
+
if (startupSyncAbort.signal.aborted) {
|
|
257
|
+
log.debug("QMD startup-sync retry: cancelled by shutdown");
|
|
258
|
+
return;
|
|
259
|
+
}
|
|
260
|
+
const synced = await orchestrator.startupSearchSync(startupSyncAbort.signal);
|
|
261
|
+
if (!synced) {
|
|
262
|
+
if (orchestrator.qmd.debugStatus() === "backend=noop") {
|
|
263
|
+
log.debug("QMD startup-sync retry: search intentionally disabled; stopping retries");
|
|
264
|
+
return;
|
|
265
|
+
}
|
|
266
|
+
log.debug(`QMD startup-sync retry: not available yet (next retry in ${RETRY_DELAYS_MS[RETRY_DELAYS_MS.indexOf(delay) + 1] ?? "n/a"}ms)`);
|
|
267
|
+
continue;
|
|
268
|
+
}
|
|
269
|
+
return;
|
|
270
|
+
}
|
|
271
|
+
log.warn("QMD startup-sync retry: exhausted all retries; search index may be stale");
|
|
272
|
+
})().catch((err) => {
|
|
273
|
+
log.warn(`QMD startup-sync retry: unexpected error: ${err}`);
|
|
274
|
+
});
|
|
275
|
+
}).catch((err) => {
|
|
276
|
+
log.warn(`Deferred init error: ${err}`);
|
|
277
|
+
});
|
|
278
|
+
return { config, service, httpServer, host, port, stop, cancelStartupSync: () => startupSyncAbort.abort(), abortDeferredInit: () => orchestrator.abortDeferredInit() };
|
|
81
279
|
}
|
|
280
|
+
var BOOLEAN_CLI_OPTIONS = /* @__PURE__ */ new Set(["help"]);
|
|
281
|
+
var VALUE_CLI_OPTIONS = /* @__PURE__ */ new Set(["config", "host", "port", "auth-token"]);
|
|
82
282
|
function parseCliArgs(argv) {
|
|
83
283
|
const args = {};
|
|
84
284
|
for (let i = 0; i < argv.length; i++) {
|
|
85
285
|
const token = argv[i];
|
|
286
|
+
if (token === "-h") {
|
|
287
|
+
args.help = "true";
|
|
288
|
+
continue;
|
|
289
|
+
}
|
|
86
290
|
if (token.startsWith("--")) {
|
|
87
|
-
const key = token.slice(2);
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
291
|
+
const [key, inlineValue] = token.slice(2).split(/=(.*)/s, 2);
|
|
292
|
+
if (!key) {
|
|
293
|
+
throw new Error(`Invalid option ${token}`);
|
|
294
|
+
}
|
|
295
|
+
if (BOOLEAN_CLI_OPTIONS.has(key)) {
|
|
296
|
+
if (inlineValue !== void 0) {
|
|
297
|
+
throw new Error(`Option --${key} does not accept a value`);
|
|
298
|
+
}
|
|
93
299
|
args[key] = "true";
|
|
300
|
+
continue;
|
|
301
|
+
}
|
|
302
|
+
if (!VALUE_CLI_OPTIONS.has(key)) {
|
|
303
|
+
throw new Error(`Unknown option --${key}`);
|
|
304
|
+
}
|
|
305
|
+
const value = inlineValue ?? argv[i + 1];
|
|
306
|
+
if (value === void 0 || inlineValue === void 0 && value.startsWith("--") || value.trim() === "") {
|
|
307
|
+
throw new Error(`Missing value for --${key}`);
|
|
94
308
|
}
|
|
309
|
+
args[key] = value;
|
|
310
|
+
if (inlineValue === void 0) i++;
|
|
95
311
|
}
|
|
96
312
|
}
|
|
97
313
|
return args;
|
|
@@ -118,27 +334,27 @@ Environment:
|
|
|
118
334
|
REMNIC_HOST Bind address (ENGRAM_HOST also supported)
|
|
119
335
|
REMNIC_AUTH_TOKEN Auth bearer token (ENGRAM_AUTH_TOKEN also supported)
|
|
120
336
|
REMNIC_MEMORY_DIR Override memory directory (ENGRAM_MEMORY_DIR also supported)
|
|
121
|
-
OPENAI_API_KEY OpenAI API key for extraction
|
|
337
|
+
OPENAI_API_KEY OpenAI API key for extraction; ignored when config sets openaiApiKey=false
|
|
122
338
|
`);
|
|
123
339
|
process.exit(0);
|
|
124
340
|
}
|
|
125
341
|
const result = await startServer({
|
|
126
342
|
configPath: args.config,
|
|
127
343
|
host: args.host,
|
|
128
|
-
port: args.port ?
|
|
344
|
+
port: args.port === void 0 ? void 0 : parseServerPort(args.port, "--port"),
|
|
129
345
|
authToken: args["auth-token"]
|
|
130
346
|
});
|
|
131
347
|
console.log(`Remnic server listening on http://${result.host}:${result.port}`);
|
|
132
348
|
const shutdown = async (signal) => {
|
|
133
349
|
console.log(`
|
|
134
350
|
Received ${signal}, shutting down...`);
|
|
135
|
-
await result.
|
|
351
|
+
await result.stop();
|
|
136
352
|
process.exit(0);
|
|
137
353
|
};
|
|
138
354
|
process.on("SIGINT", () => shutdown("SIGINT"));
|
|
139
355
|
process.on("SIGTERM", () => shutdown("SIGTERM"));
|
|
140
356
|
}
|
|
141
|
-
if (process.argv[1] && (
|
|
357
|
+
if (process.argv[1] && /(?:remnic-server|engram-server)[\\/](?:dist|src)[\\/]index\.[jt]s$/.test(process.argv[1])) {
|
|
142
358
|
cliMain().catch((err) => {
|
|
143
359
|
process.stderr.write(`Fatal: ${err instanceof Error ? err.message : String(err)}
|
|
144
360
|
`);
|
|
@@ -147,6 +363,9 @@ if (process.argv[1] && (/remnic-server[\\/](?:dist|src)[\\/]index\.[jt]s$/.test(
|
|
|
147
363
|
}
|
|
148
364
|
export {
|
|
149
365
|
cliMain,
|
|
366
|
+
loadConfigFile,
|
|
367
|
+
mergeRemnicConfigForServer,
|
|
368
|
+
parseServerConfig,
|
|
150
369
|
startServer
|
|
151
370
|
};
|
|
152
371
|
//# sourceMappingURL=index.js.map
|
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 remnic-server\n * npx remnic-server --config ./my-remnic.json\n * npx remnic-server --port 4320\n */\n\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport { parseConfig, Orchestrator, EngramAccessService, EngramAccessHttpServer, initLogger, log, getAllValidTokens, getAllValidTokensCached, 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?: number;\n authToken?: string;\n principal?: string;\n maxBodyBytes?: number;\n adminConsoleEnabled?: boolean;\n };\n}\n\nfunction readCompatEnv(primary: string, legacy: string): string | undefined {\n return process.env[primary] ?? process.env[legacy];\n}\n\nfunction resolveConfigPath(cliPath?: string): string {\n if (cliPath) return path.resolve(cliPath);\n\n const envPath = readCompatEnv(\"REMNIC_CONFIG_PATH\", \"ENGRAM_CONFIG_PATH\");\n if (envPath) return path.resolve(envPath);\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)) return candidate;\n }\n\n return path.join(homeDir, \".config\", \"remnic\", \"config.json\");\n}\n\nfunction loadConfigFile(configPath: string): ServerConfig {\n const raw = JSON.parse(fs.readFileSync(configPath, \"utf8\"));\n return {\n remnic: raw.remnic ?? raw.engram ?? raw ?? {},\n server: raw.server ?? {},\n };\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 = parseInt(port, 10);\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\n// ── Server startup ──────────────────────────────────────────────────────────\n\nexport interface ServerResult {\n config: PluginConfig;\n service: EngramAccessService;\n httpServer: EngramAccessHttpServer;\n host: string;\n port: number;\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 configPath = resolveConfigPath(options?.configPath);\n const fileConfig = fs.existsSync(configPath)\n ? loadConfigFile(configPath)\n : { remnic: {}, server: {} };\n\n const env = envOverrides();\n\n // Merge: file < env < cli flags\n const remnicConfig = { ...fileConfig.remnic, ...(env.remnic ?? {}) };\n const serverConfig = {\n ...fileConfig.server,\n ...env,\n ...(options?.host ? { host: options.host } : {}),\n ...(options?.port ? { port: options.port } : {}),\n ...(options?.authToken ? { authToken: options.authToken } : {}),\n };\n\n const config = parseConfig(remnicConfig);\n const orchestrator = new Orchestrator(config);\n const service = new EngramAccessService(orchestrator);\n\n const authToken = serverConfig.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: serverConfig.host ?? \"127.0.0.1\",\n port: serverConfig.port ?? 4318,\n authToken: authToken || undefined,\n authTokensGetter: () => getAllValidTokensCached(),\n principal: serverConfig.principal,\n maxBodyBytes: serverConfig.maxBodyBytes,\n adminConsoleEnabled: serverConfig.adminConsoleEnabled ?? false,\n citationsEnabled: config.citationsEnabled,\n citationsAutoDetect: config.citationsAutoDetect,\n });\n\n const { host, port } = await httpServer.start();\n\n return { config, service, httpServer, host, port };\n}\n\n// ── CLI entry point ──────────────────────────────────────────────────────────\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.startsWith(\"--\")) {\n const key = token.slice(2);\n const next = argv[i + 1];\n if (next && !next.startsWith(\"--\")) {\n args[key] = next;\n i++;\n } else {\n args[key] = \"true\";\n }\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\n`);\n process.exit(0);\n }\n\n const result = await startServer({\n configPath: args.config,\n host: args.host,\n port: args.port ? parseInt(args.port, 10) : undefined,\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.httpServer.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: `node .../remnic-server/dist/index.js`, `node .../remnic-server/src/index.ts`,\n// `npx remnic-server`, `npx engram-server`, but NOT test files under those directories\nif (\n process.argv[1] &&\n (/remnic-server[\\\\/](?:dist|src)[\\\\/]index\\.[jt]s$/.test(process.argv[1]) ||\n /engram-server[\\\\/](?:dist|src)[\\\\/]index\\.[jt]s$/.test(process.argv[1]) ||\n process.argv[1].endsWith(\"remnic-server\") ||\n process.argv[1].endsWith(\"engram-server\"))\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 } 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 });\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":[]}
|
package/package.json
CHANGED
|
@@ -1,19 +1,22 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@remnic/server",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "9.3.515",
|
|
4
4
|
"description": "Standalone Remnic memory server — HTTP + MCP without OpenClaw",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|
|
7
|
+
"types": "./dist/index.d.ts",
|
|
7
8
|
"exports": {
|
|
8
9
|
".": {
|
|
10
|
+
"types": "./dist/index.d.ts",
|
|
9
11
|
"import": "./dist/index.js"
|
|
10
12
|
}
|
|
11
13
|
},
|
|
12
14
|
"bin": {
|
|
13
|
-
"remnic-server": "./
|
|
14
|
-
"engram-server": "./
|
|
15
|
+
"remnic-server": "./bin/remnic-server.js",
|
|
16
|
+
"engram-server": "./bin/engram-server.js"
|
|
15
17
|
},
|
|
16
18
|
"files": [
|
|
19
|
+
"bin/*.js",
|
|
17
20
|
"dist"
|
|
18
21
|
],
|
|
19
22
|
"publishConfig": {
|
|
@@ -21,11 +24,12 @@
|
|
|
21
24
|
"provenance": true
|
|
22
25
|
},
|
|
23
26
|
"dependencies": {
|
|
24
|
-
"@remnic/core": "^
|
|
27
|
+
"@remnic/core": "^9.3.515"
|
|
25
28
|
},
|
|
26
29
|
"devDependencies": {
|
|
27
30
|
"tsup": "^8.5.1",
|
|
28
|
-
"typescript": "^5.9.3"
|
|
31
|
+
"typescript": "^5.9.3",
|
|
32
|
+
"@remnic/core": "^9.3.515"
|
|
29
33
|
},
|
|
30
34
|
"license": "MIT",
|
|
31
35
|
"repository": {
|
|
@@ -34,7 +38,9 @@
|
|
|
34
38
|
"directory": "packages/remnic-server"
|
|
35
39
|
},
|
|
36
40
|
"scripts": {
|
|
37
|
-
"build": "tsup src/index.ts --format esm --target es2022 --platform node --outDir dist &&
|
|
41
|
+
"build": "npm run check-types && tsup src/index.ts --format esm --target es2022 --platform node --outDir dist --dts && node scripts/verify-bin.mjs",
|
|
42
|
+
"verify:bin": "node scripts/verify-bin.mjs",
|
|
43
|
+
"precheck-types": "node ../../scripts/ensure-bench-build-deps.mjs",
|
|
38
44
|
"check-types": "tsc --noEmit"
|
|
39
45
|
}
|
|
40
46
|
}
|
|
@@ -1,152 +0,0 @@
|
|
|
1
|
-
// openclaw-engram: Local-first memory plugin
|
|
2
|
-
|
|
3
|
-
// src/index.ts
|
|
4
|
-
import fs from "fs";
|
|
5
|
-
import path from "path";
|
|
6
|
-
import { parseConfig, Orchestrator, EngramAccessService, EngramAccessHttpServer, initLogger, log, getAllValidTokens, getAllValidTokensCached } from "@remnic/core";
|
|
7
|
-
function readCompatEnv(primary, legacy) {
|
|
8
|
-
return process.env[primary] ?? process.env[legacy];
|
|
9
|
-
}
|
|
10
|
-
function resolveConfigPath(cliPath) {
|
|
11
|
-
if (cliPath) return path.resolve(cliPath);
|
|
12
|
-
const envPath = readCompatEnv("REMNIC_CONFIG_PATH", "ENGRAM_CONFIG_PATH");
|
|
13
|
-
if (envPath) return path.resolve(envPath);
|
|
14
|
-
const homeDir = process.env.HOME ?? "~";
|
|
15
|
-
const candidates = [
|
|
16
|
-
path.join(process.cwd(), "remnic.config.json"),
|
|
17
|
-
path.join(process.cwd(), "engram.config.json"),
|
|
18
|
-
path.join(homeDir, ".config", "remnic", "config.json"),
|
|
19
|
-
path.join(homeDir, ".config", "engram", "config.json")
|
|
20
|
-
];
|
|
21
|
-
for (const candidate of candidates) {
|
|
22
|
-
if (fs.existsSync(candidate)) return candidate;
|
|
23
|
-
}
|
|
24
|
-
return path.join(homeDir, ".config", "remnic", "config.json");
|
|
25
|
-
}
|
|
26
|
-
function loadConfigFile(configPath) {
|
|
27
|
-
const raw = JSON.parse(fs.readFileSync(configPath, "utf8"));
|
|
28
|
-
return {
|
|
29
|
-
remnic: raw.remnic ?? raw.engram ?? raw ?? {},
|
|
30
|
-
server: raw.server ?? {}
|
|
31
|
-
};
|
|
32
|
-
}
|
|
33
|
-
function envOverrides() {
|
|
34
|
-
const overrides = {};
|
|
35
|
-
const remnic = {};
|
|
36
|
-
const port = readCompatEnv("REMNIC_PORT", "ENGRAM_PORT");
|
|
37
|
-
const host = readCompatEnv("REMNIC_HOST", "ENGRAM_HOST");
|
|
38
|
-
const authToken = readCompatEnv("REMNIC_AUTH_TOKEN", "ENGRAM_AUTH_TOKEN");
|
|
39
|
-
if (port) overrides.port = parseInt(port, 10);
|
|
40
|
-
if (host) overrides.host = host;
|
|
41
|
-
if (authToken) overrides.authToken = authToken;
|
|
42
|
-
if (process.env.OPENAI_API_KEY) remnic.openaiApiKey = process.env.OPENAI_API_KEY;
|
|
43
|
-
const memoryDir = readCompatEnv("REMNIC_MEMORY_DIR", "ENGRAM_MEMORY_DIR");
|
|
44
|
-
if (memoryDir) remnic.memoryDir = memoryDir;
|
|
45
|
-
return { ...overrides, ...Object.keys(remnic).length > 0 ? { remnic } : {} };
|
|
46
|
-
}
|
|
47
|
-
async function startServer(options) {
|
|
48
|
-
initLogger();
|
|
49
|
-
const configPath = resolveConfigPath(options?.configPath);
|
|
50
|
-
const fileConfig = fs.existsSync(configPath) ? loadConfigFile(configPath) : { remnic: {}, server: {} };
|
|
51
|
-
const env = envOverrides();
|
|
52
|
-
const remnicConfig = { ...fileConfig.remnic, ...env.remnic ?? {} };
|
|
53
|
-
const serverConfig = {
|
|
54
|
-
...fileConfig.server,
|
|
55
|
-
...env,
|
|
56
|
-
...options?.host ? { host: options.host } : {},
|
|
57
|
-
...options?.port ? { port: options.port } : {},
|
|
58
|
-
...options?.authToken ? { authToken: options.authToken } : {}
|
|
59
|
-
};
|
|
60
|
-
const config = parseConfig(remnicConfig);
|
|
61
|
-
const orchestrator = new Orchestrator(config);
|
|
62
|
-
const service = new EngramAccessService(orchestrator);
|
|
63
|
-
const authToken = serverConfig.authToken ?? readCompatEnv("REMNIC_AUTH_TOKEN", "ENGRAM_AUTH_TOKEN") ?? "";
|
|
64
|
-
if (!authToken && getAllValidTokens().length === 0) {
|
|
65
|
-
log.warn("No auth token set \u2014 server will reject all requests. Set REMNIC_AUTH_TOKEN, server.authToken in config, or generate tokens with 'remnic token generate'.");
|
|
66
|
-
}
|
|
67
|
-
const httpServer = new EngramAccessHttpServer({
|
|
68
|
-
service,
|
|
69
|
-
host: serverConfig.host ?? "127.0.0.1",
|
|
70
|
-
port: serverConfig.port ?? 4318,
|
|
71
|
-
authToken: authToken || void 0,
|
|
72
|
-
authTokensGetter: () => getAllValidTokensCached(),
|
|
73
|
-
principal: serverConfig.principal,
|
|
74
|
-
maxBodyBytes: serverConfig.maxBodyBytes,
|
|
75
|
-
adminConsoleEnabled: serverConfig.adminConsoleEnabled ?? false,
|
|
76
|
-
citationsEnabled: config.citationsEnabled,
|
|
77
|
-
citationsAutoDetect: config.citationsAutoDetect
|
|
78
|
-
});
|
|
79
|
-
const { host, port } = await httpServer.start();
|
|
80
|
-
return { config, service, httpServer, host, port };
|
|
81
|
-
}
|
|
82
|
-
function parseCliArgs(argv) {
|
|
83
|
-
const args = {};
|
|
84
|
-
for (let i = 0; i < argv.length; i++) {
|
|
85
|
-
const token = argv[i];
|
|
86
|
-
if (token.startsWith("--")) {
|
|
87
|
-
const key = token.slice(2);
|
|
88
|
-
const next = argv[i + 1];
|
|
89
|
-
if (next && !next.startsWith("--")) {
|
|
90
|
-
args[key] = next;
|
|
91
|
-
i++;
|
|
92
|
-
} else {
|
|
93
|
-
args[key] = "true";
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
}
|
|
97
|
-
return args;
|
|
98
|
-
}
|
|
99
|
-
async function cliMain(argv = process.argv.slice(2)) {
|
|
100
|
-
const args = parseCliArgs(argv);
|
|
101
|
-
if (args.help) {
|
|
102
|
-
console.log(`
|
|
103
|
-
remnic-server \u2014 Standalone Remnic memory server
|
|
104
|
-
|
|
105
|
-
Usage:
|
|
106
|
-
remnic-server [options]
|
|
107
|
-
|
|
108
|
-
Options:
|
|
109
|
-
--config <path> Path to config file (default: remnic.config.json)
|
|
110
|
-
--host <addr> Bind address (default: 127.0.0.1)
|
|
111
|
-
--port <number> Port number (default: 4318)
|
|
112
|
-
--auth-token <tok> Bearer token for auth (or set REMNIC_AUTH_TOKEN)
|
|
113
|
-
--help Show this help
|
|
114
|
-
|
|
115
|
-
Environment:
|
|
116
|
-
REMNIC_CONFIG_PATH Config file path (ENGRAM_CONFIG_PATH also supported)
|
|
117
|
-
REMNIC_PORT Server port (ENGRAM_PORT also supported)
|
|
118
|
-
REMNIC_HOST Bind address (ENGRAM_HOST also supported)
|
|
119
|
-
REMNIC_AUTH_TOKEN Auth bearer token (ENGRAM_AUTH_TOKEN also supported)
|
|
120
|
-
REMNIC_MEMORY_DIR Override memory directory (ENGRAM_MEMORY_DIR also supported)
|
|
121
|
-
OPENAI_API_KEY OpenAI API key for extraction
|
|
122
|
-
`);
|
|
123
|
-
process.exit(0);
|
|
124
|
-
}
|
|
125
|
-
const result = await startServer({
|
|
126
|
-
configPath: args.config,
|
|
127
|
-
host: args.host,
|
|
128
|
-
port: args.port ? parseInt(args.port, 10) : void 0,
|
|
129
|
-
authToken: args["auth-token"]
|
|
130
|
-
});
|
|
131
|
-
console.log(`Remnic server listening on http://${result.host}:${result.port}`);
|
|
132
|
-
const shutdown = async (signal) => {
|
|
133
|
-
console.log(`
|
|
134
|
-
Received ${signal}, shutting down...`);
|
|
135
|
-
await result.httpServer.stop();
|
|
136
|
-
process.exit(0);
|
|
137
|
-
};
|
|
138
|
-
process.on("SIGINT", () => shutdown("SIGINT"));
|
|
139
|
-
process.on("SIGTERM", () => shutdown("SIGTERM"));
|
|
140
|
-
}
|
|
141
|
-
if (process.argv[1] && (/remnic-server[\\/](?:dist|src)[\\/]index\.[jt]s$/.test(process.argv[1]) || /engram-server[\\/](?:dist|src)[\\/]index\.[jt]s$/.test(process.argv[1]) || process.argv[1].endsWith("remnic-server") || process.argv[1].endsWith("engram-server"))) {
|
|
142
|
-
cliMain().catch((err) => {
|
|
143
|
-
process.stderr.write(`Fatal: ${err instanceof Error ? err.message : String(err)}
|
|
144
|
-
`);
|
|
145
|
-
process.exit(1);
|
|
146
|
-
});
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
export {
|
|
150
|
-
cliMain
|
|
151
|
-
};
|
|
152
|
-
//# sourceMappingURL=chunk-MKNXEH3P.js.map
|
|
@@ -1 +0,0 @@
|
|
|
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 remnic-server\n * npx remnic-server --config ./my-remnic.json\n * npx remnic-server --port 4320\n */\n\nimport fs from \"node:fs\";\nimport path from \"node:path\";\nimport { parseConfig, Orchestrator, EngramAccessService, EngramAccessHttpServer, initLogger, log, getAllValidTokens, getAllValidTokensCached, 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?: number;\n authToken?: string;\n principal?: string;\n maxBodyBytes?: number;\n adminConsoleEnabled?: boolean;\n };\n}\n\nfunction readCompatEnv(primary: string, legacy: string): string | undefined {\n return process.env[primary] ?? process.env[legacy];\n}\n\nfunction resolveConfigPath(cliPath?: string): string {\n if (cliPath) return path.resolve(cliPath);\n\n const envPath = readCompatEnv(\"REMNIC_CONFIG_PATH\", \"ENGRAM_CONFIG_PATH\");\n if (envPath) return path.resolve(envPath);\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)) return candidate;\n }\n\n return path.join(homeDir, \".config\", \"remnic\", \"config.json\");\n}\n\nfunction loadConfigFile(configPath: string): ServerConfig {\n const raw = JSON.parse(fs.readFileSync(configPath, \"utf8\"));\n return {\n remnic: raw.remnic ?? raw.engram ?? raw ?? {},\n server: raw.server ?? {},\n };\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 = parseInt(port, 10);\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\n// ── Server startup ──────────────────────────────────────────────────────────\n\nexport interface ServerResult {\n config: PluginConfig;\n service: EngramAccessService;\n httpServer: EngramAccessHttpServer;\n host: string;\n port: number;\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 configPath = resolveConfigPath(options?.configPath);\n const fileConfig = fs.existsSync(configPath)\n ? loadConfigFile(configPath)\n : { remnic: {}, server: {} };\n\n const env = envOverrides();\n\n // Merge: file < env < cli flags\n const remnicConfig = { ...fileConfig.remnic, ...(env.remnic ?? {}) };\n const serverConfig = {\n ...fileConfig.server,\n ...env,\n ...(options?.host ? { host: options.host } : {}),\n ...(options?.port ? { port: options.port } : {}),\n ...(options?.authToken ? { authToken: options.authToken } : {}),\n };\n\n const config = parseConfig(remnicConfig);\n const orchestrator = new Orchestrator(config);\n const service = new EngramAccessService(orchestrator);\n\n const authToken = serverConfig.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: serverConfig.host ?? \"127.0.0.1\",\n port: serverConfig.port ?? 4318,\n authToken: authToken || undefined,\n authTokensGetter: () => getAllValidTokensCached(),\n principal: serverConfig.principal,\n maxBodyBytes: serverConfig.maxBodyBytes,\n adminConsoleEnabled: serverConfig.adminConsoleEnabled ?? false,\n citationsEnabled: config.citationsEnabled,\n citationsAutoDetect: config.citationsAutoDetect,\n });\n\n const { host, port } = await httpServer.start();\n\n return { config, service, httpServer, host, port };\n}\n\n// ── CLI entry point ──────────────────────────────────────────────────────────\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.startsWith(\"--\")) {\n const key = token.slice(2);\n const next = argv[i + 1];\n if (next && !next.startsWith(\"--\")) {\n args[key] = next;\n i++;\n } else {\n args[key] = \"true\";\n }\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\n`);\n process.exit(0);\n }\n\n const result = await startServer({\n configPath: args.config,\n host: args.host,\n port: args.port ? parseInt(args.port, 10) : undefined,\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.httpServer.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: `node .../remnic-server/dist/index.js`, `node .../remnic-server/src/index.ts`,\n// `npx remnic-server`, `npx engram-server`, but NOT test files under those directories\nif (\n process.argv[1] &&\n (/remnic-server[\\\\/](?:dist|src)[\\\\/]index\\.[jt]s$/.test(process.argv[1]) ||\n /engram-server[\\\\/](?:dist|src)[\\\\/]index\\.[jt]s$/.test(process.argv[1]) ||\n process.argv[1].endsWith(\"remnic-server\") ||\n process.argv[1].endsWith(\"engram-server\"))\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,13 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
// openclaw-engram: Local-first memory plugin
|
|
3
|
-
import {
|
|
4
|
-
cliMain
|
|
5
|
-
} from "./chunk-MKNXEH3P.js";
|
|
6
|
-
|
|
7
|
-
// bin/engram-server.ts
|
|
8
|
-
cliMain().catch((err) => {
|
|
9
|
-
process.stderr.write(`Fatal: ${err instanceof Error ? err.message : String(err)}
|
|
10
|
-
`);
|
|
11
|
-
process.exit(1);
|
|
12
|
-
});
|
|
13
|
-
//# sourceMappingURL=engram-server.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../../bin/engram-server.ts"],"sourcesContent":["#!/usr/bin/env node\n/**\n * engram-server binary entry point.\n * Delegates to @remnic/server CLI main.\n */\nimport { cliMain } from \"../src/index.js\";\n\ncliMain().catch((err) => {\n process.stderr.write(`Fatal: ${err instanceof Error ? err.message : String(err)}\\n`);\n process.exit(1);\n});\n"],"mappings":";;;;;;;AAOA,QAAQ,EAAE,MAAM,CAAC,QAAQ;AACvB,UAAQ,OAAO,MAAM,UAAU,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC;AAAA,CAAI;AACnF,UAAQ,KAAK,CAAC;AAChB,CAAC;","names":[]}
|
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
// openclaw-engram: Local-first memory plugin
|
|
3
|
-
import {
|
|
4
|
-
cliMain
|
|
5
|
-
} from "./chunk-MKNXEH3P.js";
|
|
6
|
-
|
|
7
|
-
// bin/remnic-server.ts
|
|
8
|
-
cliMain().catch((err) => {
|
|
9
|
-
process.stderr.write(`Fatal: ${err instanceof Error ? err.message : String(err)}
|
|
10
|
-
`);
|
|
11
|
-
process.exit(1);
|
|
12
|
-
});
|
|
13
|
-
//# sourceMappingURL=remnic-server.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../../bin/remnic-server.ts"],"sourcesContent":["#!/usr/bin/env node\n/**\n * remnic-server binary entry point.\n * Delegates to @remnic/server CLI main.\n */\nimport { cliMain } from \"../src/index.js\";\n\ncliMain().catch((err) => {\n process.stderr.write(`Fatal: ${err instanceof Error ? err.message : String(err)}\\n`);\n process.exit(1);\n});\n"],"mappings":";;;;;;;AAOA,QAAQ,EAAE,MAAM,CAAC,QAAQ;AACvB,UAAQ,OAAO,MAAM,UAAU,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC;AAAA,CAAI;AACnF,UAAQ,KAAK,CAAC;AAChB,CAAC;","names":[]}
|