@sochdb/sochdb 0.4.0

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.
Files changed (78) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +3349 -0
  3. package/_bin/aarch64-apple-darwin/libsochdb_storage.dylib +0 -0
  4. package/_bin/aarch64-apple-darwin/sochdb-bulk +0 -0
  5. package/_bin/aarch64-apple-darwin/sochdb-grpc-server +0 -0
  6. package/_bin/aarch64-apple-darwin/sochdb-server +0 -0
  7. package/_bin/x86_64-pc-windows-msvc/sochdb-bulk.exe +0 -0
  8. package/_bin/x86_64-pc-windows-msvc/sochdb-grpc-server.exe +0 -0
  9. package/_bin/x86_64-pc-windows-msvc/sochdb_storage.dll +0 -0
  10. package/_bin/x86_64-unknown-linux-gnu/libsochdb_storage.so +0 -0
  11. package/_bin/x86_64-unknown-linux-gnu/sochdb-bulk +0 -0
  12. package/_bin/x86_64-unknown-linux-gnu/sochdb-grpc-server +0 -0
  13. package/_bin/x86_64-unknown-linux-gnu/sochdb-server +0 -0
  14. package/bin/sochdb-bulk.js +80 -0
  15. package/bin/sochdb-grpc-server.js +80 -0
  16. package/bin/sochdb-server.js +84 -0
  17. package/dist/cjs/analytics.js +196 -0
  18. package/dist/cjs/database.js +929 -0
  19. package/dist/cjs/embedded/database.js +236 -0
  20. package/dist/cjs/embedded/ffi/bindings.js +113 -0
  21. package/dist/cjs/embedded/ffi/library-finder.js +135 -0
  22. package/dist/cjs/embedded/index.js +14 -0
  23. package/dist/cjs/embedded/transaction.js +172 -0
  24. package/dist/cjs/errors.js +71 -0
  25. package/dist/cjs/format.js +176 -0
  26. package/dist/cjs/grpc-client.js +328 -0
  27. package/dist/cjs/index.js +75 -0
  28. package/dist/cjs/ipc-client.js +504 -0
  29. package/dist/cjs/query.js +154 -0
  30. package/dist/cjs/server-manager.js +295 -0
  31. package/dist/cjs/sql-engine.js +874 -0
  32. package/dist/esm/analytics.js +196 -0
  33. package/dist/esm/database.js +931 -0
  34. package/dist/esm/embedded/database.js +239 -0
  35. package/dist/esm/embedded/ffi/bindings.js +142 -0
  36. package/dist/esm/embedded/ffi/library-finder.js +135 -0
  37. package/dist/esm/embedded/index.js +14 -0
  38. package/dist/esm/embedded/transaction.js +176 -0
  39. package/dist/esm/errors.js +71 -0
  40. package/dist/esm/format.js +179 -0
  41. package/dist/esm/grpc-client.js +333 -0
  42. package/dist/esm/index.js +75 -0
  43. package/dist/esm/ipc-client.js +505 -0
  44. package/dist/esm/query.js +159 -0
  45. package/dist/esm/server-manager.js +295 -0
  46. package/dist/esm/sql-engine.js +875 -0
  47. package/dist/types/analytics.d.ts +66 -0
  48. package/dist/types/analytics.d.ts.map +1 -0
  49. package/dist/types/database.d.ts +523 -0
  50. package/dist/types/database.d.ts.map +1 -0
  51. package/dist/types/embedded/database.d.ts +105 -0
  52. package/dist/types/embedded/database.d.ts.map +1 -0
  53. package/dist/types/embedded/ffi/bindings.d.ts +24 -0
  54. package/dist/types/embedded/ffi/bindings.d.ts.map +1 -0
  55. package/dist/types/embedded/ffi/library-finder.d.ts +17 -0
  56. package/dist/types/embedded/ffi/library-finder.d.ts.map +1 -0
  57. package/dist/types/embedded/index.d.ts +9 -0
  58. package/dist/types/embedded/index.d.ts.map +1 -0
  59. package/dist/types/embedded/transaction.d.ts +21 -0
  60. package/dist/types/embedded/transaction.d.ts.map +1 -0
  61. package/dist/types/errors.d.ts +36 -0
  62. package/dist/types/errors.d.ts.map +1 -0
  63. package/dist/types/format.d.ts +117 -0
  64. package/dist/types/format.d.ts.map +1 -0
  65. package/dist/types/grpc-client.d.ts +120 -0
  66. package/dist/types/grpc-client.d.ts.map +1 -0
  67. package/dist/types/index.d.ts +50 -0
  68. package/dist/types/index.d.ts.map +1 -0
  69. package/dist/types/ipc-client.d.ts +177 -0
  70. package/dist/types/ipc-client.d.ts.map +1 -0
  71. package/dist/types/query.d.ts +85 -0
  72. package/dist/types/query.d.ts.map +1 -0
  73. package/dist/types/server-manager.d.ts +29 -0
  74. package/dist/types/server-manager.d.ts.map +1 -0
  75. package/dist/types/sql-engine.d.ts +100 -0
  76. package/dist/types/sql-engine.d.ts.map +1 -0
  77. package/package.json +90 -0
  78. package/scripts/postinstall.js +50 -0
@@ -0,0 +1,159 @@
1
+ "use strict";
2
+ /**
3
+ * SochDB Query Builder
4
+ *
5
+ * Fluent query interface for SochDB.
6
+ *
7
+ * @packageDocumentation
8
+ */
9
+ Object.defineProperty(exports, "__esModule", { value: true });
10
+ exports.Query = void 0;
11
+ /**
12
+ * Fluent query builder for SochDB.
13
+ *
14
+ * @example
15
+ * ```typescript
16
+ * const results = await db.query('users/')
17
+ * .limit(10)
18
+ * .select(['name', 'email'])
19
+ * .toList();
20
+ * ```
21
+ */
22
+ class Query {
23
+ _client;
24
+ _pathPrefix;
25
+ _limit;
26
+ _offset;
27
+ _columns;
28
+ constructor(client, pathPrefix) {
29
+ this._client = client;
30
+ this._pathPrefix = pathPrefix;
31
+ }
32
+ /**
33
+ * Limit the number of results.
34
+ *
35
+ * @param n - Maximum number of results to return
36
+ * @returns This query builder for chaining
37
+ */
38
+ limit(n) {
39
+ this._limit = n;
40
+ return this;
41
+ }
42
+ /**
43
+ * Skip the first n results.
44
+ *
45
+ * @param n - Number of results to skip
46
+ * @returns This query builder for chaining
47
+ */
48
+ offset(n) {
49
+ this._offset = n;
50
+ return this;
51
+ }
52
+ /**
53
+ * Select specific columns to return.
54
+ *
55
+ * @param columns - Array of column names to select
56
+ * @returns This query builder for chaining
57
+ */
58
+ select(columns) {
59
+ this._columns = columns;
60
+ return this;
61
+ }
62
+ /**
63
+ * Execute the query and return results as TOON string.
64
+ *
65
+ * @returns TOON formatted string (e.g., "result[N]{cols}: row1; row2")
66
+ */
67
+ async execute() {
68
+ return this._client.query(this._pathPrefix, {
69
+ limit: this._limit,
70
+ offset: this._offset,
71
+ columns: this._columns,
72
+ });
73
+ }
74
+ /**
75
+ * Execute and parse results into a list of objects.
76
+ *
77
+ * @returns Array of result objects
78
+ */
79
+ async toList() {
80
+ const toonStr = await this.execute();
81
+ return this._parseToon(toonStr);
82
+ }
83
+ /**
84
+ * Execute and return the first result, or null if none.
85
+ *
86
+ * @returns First result or null
87
+ */
88
+ async first() {
89
+ const originalLimit = this._limit;
90
+ this._limit = 1;
91
+ const results = await this.toList();
92
+ this._limit = originalLimit;
93
+ return results.length > 0 ? results[0] : null;
94
+ }
95
+ /**
96
+ * Execute and return the count of results.
97
+ *
98
+ * @returns Number of matching results
99
+ */
100
+ async count() {
101
+ const results = await this.toList();
102
+ return results.length;
103
+ }
104
+ /**
105
+ * Simple TOON parser.
106
+ *
107
+ * Parses TOON format: "result[N]{col1,col2}: val1,val2; val3,val4"
108
+ */
109
+ _parseToon(toonStr) {
110
+ if (!toonStr || toonStr === 'result[0]{}:') {
111
+ return [];
112
+ }
113
+ // Parse header: result[N]{cols}:
114
+ const headerMatch = toonStr.match(/^result\[(\d+)\]\{([^}]*)\}:\s*/);
115
+ if (!headerMatch) {
116
+ // Try to parse as JSON if not TOON format
117
+ try {
118
+ return JSON.parse(toonStr);
119
+ }
120
+ catch {
121
+ return [];
122
+ }
123
+ }
124
+ const count = parseInt(headerMatch[1], 10);
125
+ if (count === 0) {
126
+ return [];
127
+ }
128
+ const columns = headerMatch[2].split(',').map((c) => c.trim());
129
+ const body = toonStr.substring(headerMatch[0].length);
130
+ // Split rows by semicolon
131
+ const rows = body.split(';').map((r) => r.trim()).filter((r) => r.length > 0);
132
+ return rows.map((row) => {
133
+ const values = row.split(',').map((v) => v.trim());
134
+ const result = {};
135
+ columns.forEach((col, idx) => {
136
+ if (col && idx < values.length) {
137
+ // Try to parse as JSON/number
138
+ let value = values[idx];
139
+ if (value === 'null') {
140
+ value = null;
141
+ }
142
+ else if (value === 'true') {
143
+ value = true;
144
+ }
145
+ else if (value === 'false') {
146
+ value = false;
147
+ }
148
+ else if (!isNaN(Number(value))) {
149
+ value = Number(value);
150
+ }
151
+ result[col] = value;
152
+ }
153
+ });
154
+ return result;
155
+ });
156
+ }
157
+ }
158
+ exports.Query = Query;
159
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query.js","sourceRoot":"","sources":["../../src/query.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SochDB Query Builder\n *\n * Fluent query interface for SochDB.\n *\n * @packageDocumentation\n */\n\n// Copyright 2025 Sushanth (https://github.com/sushanthpy)\n//\n// Licensed under the Apache License, Version 2.0 (the \"License\");\n// you may not use this file except in compliance with the License.\n// You may obtain a copy of the License at\n//\n//     http://www.apache.org/licenses/LICENSE-2.0\n\nimport { IpcClient } from './ipc-client';\n\n/**\n * Query result row.\n */\nexport interface QueryResult {\n  [key: string]: unknown;\n}\n\n/**\n * Fluent query builder for SochDB.\n *\n * @example\n * ```typescript\n * const results = await db.query('users/')\n *   .limit(10)\n *   .select(['name', 'email'])\n *   .toList();\n * ```\n */\nexport class Query {\n  private _client: IpcClient;\n  private _pathPrefix: string;\n  private _limit?: number;\n  private _offset?: number;\n  private _columns?: string[];\n\n  constructor(client: IpcClient, pathPrefix: string) {\n    this._client = client;\n    this._pathPrefix = pathPrefix;\n  }\n\n  /**\n   * Limit the number of results.\n   *\n   * @param n - Maximum number of results to return\n   * @returns This query builder for chaining\n   */\n  limit(n: number): Query {\n    this._limit = n;\n    return this;\n  }\n\n  /**\n   * Skip the first n results.\n   *\n   * @param n - Number of results to skip\n   * @returns This query builder for chaining\n   */\n  offset(n: number): Query {\n    this._offset = n;\n    return this;\n  }\n\n  /**\n   * Select specific columns to return.\n   *\n   * @param columns - Array of column names to select\n   * @returns This query builder for chaining\n   */\n  select(columns: string[]): Query {\n    this._columns = columns;\n    return this;\n  }\n\n  /**\n   * Execute the query and return results as TOON string.\n   *\n   * @returns TOON formatted string (e.g., \"result[N]{cols}: row1; row2\")\n   */\n  async execute(): Promise<string> {\n    return this._client.query(this._pathPrefix, {\n      limit: this._limit,\n      offset: this._offset,\n      columns: this._columns,\n    });\n  }\n\n  /**\n   * Execute and parse results into a list of objects.\n   *\n   * @returns Array of result objects\n   */\n  async toList(): Promise<QueryResult[]> {\n    const toonStr = await this.execute();\n    return this._parseToon(toonStr);\n  }\n\n  /**\n   * Execute and return the first result, or null if none.\n   *\n   * @returns First result or null\n   */\n  async first(): Promise<QueryResult | null> {\n    const originalLimit = this._limit;\n    this._limit = 1;\n    const results = await this.toList();\n    this._limit = originalLimit;\n    return results.length > 0 ? results[0] : null;\n  }\n\n  /**\n   * Execute and return the count of results.\n   *\n   * @returns Number of matching results\n   */\n  async count(): Promise<number> {\n    const results = await this.toList();\n    return results.length;\n  }\n\n  /**\n   * Simple TOON parser.\n   *\n   * Parses TOON format: \"result[N]{col1,col2}: val1,val2; val3,val4\"\n   */\n  private _parseToon(toonStr: string): QueryResult[] {\n    if (!toonStr || toonStr === 'result[0]{}:') {\n      return [];\n    }\n\n    // Parse header: result[N]{cols}:\n    const headerMatch = toonStr.match(/^result\\[(\\d+)\\]\\{([^}]*)\\}:\\s*/);\n    if (!headerMatch) {\n      // Try to parse as JSON if not TOON format\n      try {\n        return JSON.parse(toonStr);\n      } catch {\n        return [];\n      }\n    }\n\n    const count = parseInt(headerMatch[1], 10);\n    if (count === 0) {\n      return [];\n    }\n\n    const columns = headerMatch[2].split(',').map((c) => c.trim());\n    const body = toonStr.substring(headerMatch[0].length);\n\n    // Split rows by semicolon\n    const rows = body.split(';').map((r) => r.trim()).filter((r) => r.length > 0);\n\n    return rows.map((row) => {\n      const values = row.split(',').map((v) => v.trim());\n      const result: QueryResult = {};\n\n      columns.forEach((col, idx) => {\n        if (col && idx < values.length) {\n          // Try to parse as JSON/number\n          let value: unknown = values[idx];\n          if (value === 'null') {\n            value = null;\n          } else if (value === 'true') {\n            value = true;\n          } else if (value === 'false') {\n            value = false;\n          } else if (!isNaN(Number(value))) {\n            value = Number(value);\n          }\n          result[col] = value;\n        }\n      });\n\n      return result;\n    });\n  }\n}\n"]}
@@ -0,0 +1,295 @@
1
+ "use strict";
2
+ /**
3
+ * SochDB Embedded Server Manager
4
+ *
5
+ * Manages the lifecycle of the SochDB server process for embedded mode.
6
+ * Automatically starts the server when needed and stops it on cleanup.
7
+ *
8
+ * @packageDocumentation
9
+ */
10
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
11
+ if (k2 === undefined) k2 = k;
12
+ var desc = Object.getOwnPropertyDescriptor(m, k);
13
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
14
+ desc = { enumerable: true, get: function() { return m[k]; } };
15
+ }
16
+ Object.defineProperty(o, k2, desc);
17
+ }) : (function(o, m, k, k2) {
18
+ if (k2 === undefined) k2 = k;
19
+ o[k2] = m[k];
20
+ }));
21
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
22
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
23
+ }) : function(o, v) {
24
+ o["default"] = v;
25
+ });
26
+ var __importStar = (this && this.__importStar) || (function () {
27
+ var ownKeys = function(o) {
28
+ ownKeys = Object.getOwnPropertyNames || function (o) {
29
+ var ar = [];
30
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
31
+ return ar;
32
+ };
33
+ return ownKeys(o);
34
+ };
35
+ return function (mod) {
36
+ if (mod && mod.__esModule) return mod;
37
+ var result = {};
38
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
39
+ __setModuleDefault(result, mod);
40
+ return result;
41
+ };
42
+ })();
43
+ Object.defineProperty(exports, "__esModule", { value: true });
44
+ exports.startEmbeddedServer = startEmbeddedServer;
45
+ exports.stopEmbeddedServer = stopEmbeddedServer;
46
+ exports.stopAllEmbeddedServers = stopAllEmbeddedServers;
47
+ exports.isServerRunning = isServerRunning;
48
+ // Copyright 2025 Sushanth (https://github.com/sushanthpy)
49
+ //
50
+ // Licensed under the Apache License, Version 2.0 (the "License");
51
+ // you may not use this file except in compliance with the License.
52
+ const fs = __importStar(require("fs"));
53
+ const path = __importStar(require("path"));
54
+ const net = __importStar(require("net"));
55
+ const child_process_1 = require("child_process");
56
+ const errors_1 = require("./errors");
57
+ /**
58
+ * Find the sochdb-server binary (provides IPC interface)
59
+ *
60
+ * Note: sochdb-server uses Unix domain sockets, not available on Windows.
61
+ * On Windows, use the gRPC client instead for cross-platform compatibility.
62
+ */
63
+ function findServerBinary() {
64
+ const platform = process.platform;
65
+ const arch = process.arch;
66
+ // Windows doesn't support Unix sockets, sochdb-server is not available
67
+ if (platform === 'win32') {
68
+ throw new errors_1.DatabaseError('sochdb-server is not available on Windows (requires Unix domain sockets). ' +
69
+ 'Use the gRPC client for cross-platform support: ' +
70
+ 'const client = await GrpcClient.connect("localhost:50051")');
71
+ }
72
+ let target;
73
+ if (platform === 'darwin') {
74
+ target = arch === 'arm64' ? 'aarch64-apple-darwin' : 'x86_64-apple-darwin';
75
+ }
76
+ else {
77
+ target = arch === 'arm64' ? 'aarch64-unknown-linux-gnu' : 'x86_64-unknown-linux-gnu';
78
+ }
79
+ const binaryName = 'sochdb-server';
80
+ // Search paths - prioritize bundled binaries
81
+ const searchPaths = [
82
+ // Bundled in package (installed via npm) - from dist/cjs or dist/esm
83
+ path.join(__dirname, '..', '_bin', target, binaryName),
84
+ path.join(__dirname, '..', '..', '_bin', target, binaryName),
85
+ path.join(__dirname, '..', '..', '..', '_bin', target, binaryName),
86
+ // When running from source (src/) during development/testing
87
+ path.resolve(__dirname, '..', '_bin', target, binaryName),
88
+ // Development paths - from project root
89
+ path.join(__dirname, '..', '..', 'target', 'release', binaryName),
90
+ path.join(__dirname, '..', '..', 'target', 'debug', binaryName),
91
+ path.join(__dirname, '..', '..', '..', 'target', 'release', binaryName),
92
+ path.join(__dirname, '..', '..', '..', 'target', 'debug', binaryName),
93
+ // Absolute paths for sochdb workspace
94
+ path.resolve(process.cwd(), '_bin', target, binaryName),
95
+ path.resolve(process.cwd(), 'target', 'release', binaryName),
96
+ path.resolve(process.cwd(), '..', 'target', 'release', binaryName),
97
+ ];
98
+ for (const p of searchPaths) {
99
+ if (fs.existsSync(p)) {
100
+ return p;
101
+ }
102
+ }
103
+ // Try PATH
104
+ const pathDirs = (process.env.PATH || '').split(path.delimiter);
105
+ for (const dir of pathDirs) {
106
+ const p = path.join(dir, binaryName);
107
+ if (fs.existsSync(p)) {
108
+ return p;
109
+ }
110
+ }
111
+ throw new errors_1.DatabaseError(`Could not find ${binaryName}. ` +
112
+ `The pre-built binary may not be available for your platform (${platform}/${arch}). ` +
113
+ `Install via: cargo build --release -p sochdb-tools`);
114
+ }
115
+ /**
116
+ * Wait for a Unix socket to become available
117
+ */
118
+ async function waitForSocket(socketPath, timeoutMs = 10000) {
119
+ const startTime = Date.now();
120
+ const checkInterval = 100;
121
+ while (Date.now() - startTime < timeoutMs) {
122
+ if (fs.existsSync(socketPath)) {
123
+ // Try to connect to verify it's actually listening
124
+ try {
125
+ await new Promise((resolve, reject) => {
126
+ const socket = net.createConnection({ path: socketPath }, () => {
127
+ socket.destroy();
128
+ resolve();
129
+ });
130
+ socket.on('error', reject);
131
+ socket.setTimeout(1000);
132
+ });
133
+ return;
134
+ }
135
+ catch {
136
+ // Socket exists but not ready yet
137
+ }
138
+ }
139
+ await new Promise(resolve => setTimeout(resolve, checkInterval));
140
+ }
141
+ throw new errors_1.ConnectionError(`Timeout waiting for server socket at ${socketPath} after ${timeoutMs}ms`);
142
+ }
143
+ // Track running server instances
144
+ const runningServers = new Map();
145
+ /**
146
+ * Start an embedded SochDB server for the given database path.
147
+ * If a server is already running for this path, return the existing instance.
148
+ *
149
+ * @param dbPath - Path to the database directory
150
+ * @returns The socket path for connecting
151
+ */
152
+ async function startEmbeddedServer(dbPath) {
153
+ const absolutePath = path.resolve(dbPath);
154
+ const socketPath = path.join(absolutePath, 'sochdb.sock');
155
+ // Check if server already running for this path
156
+ const existing = runningServers.get(absolutePath);
157
+ if (existing && existing.process.exitCode === null) {
158
+ // Server still running
159
+ return socketPath;
160
+ }
161
+ // Check if another process already has a server running
162
+ if (fs.existsSync(socketPath)) {
163
+ try {
164
+ // Try to connect - if successful, server is running
165
+ await waitForSocket(socketPath, 1000);
166
+ return socketPath;
167
+ }
168
+ catch {
169
+ // Socket exists but dead - clean it up
170
+ try {
171
+ fs.unlinkSync(socketPath);
172
+ }
173
+ catch {
174
+ // Ignore
175
+ }
176
+ }
177
+ }
178
+ // Ensure database directory exists
179
+ if (!fs.existsSync(absolutePath)) {
180
+ fs.mkdirSync(absolutePath, { recursive: true });
181
+ }
182
+ // Find and spawn server
183
+ const serverBinary = findServerBinary();
184
+ const serverProcess = (0, child_process_1.spawn)(serverBinary, ['--db', absolutePath, '--socket', socketPath], {
185
+ stdio: ['ignore', 'pipe', 'pipe'],
186
+ detached: false,
187
+ });
188
+ // Collect stderr for error reporting
189
+ let stderrOutput = '';
190
+ serverProcess.stderr?.on('data', (data) => {
191
+ stderrOutput += data.toString();
192
+ });
193
+ // Handle process exit
194
+ serverProcess.on('exit', (code, signal) => {
195
+ runningServers.delete(absolutePath);
196
+ if (code !== 0 && code !== null) {
197
+ console.error(`SochDB server exited with code ${code}: ${stderrOutput}`);
198
+ }
199
+ });
200
+ serverProcess.on('error', (err) => {
201
+ runningServers.delete(absolutePath);
202
+ console.error(`Failed to start SochDB server: ${err.message}`);
203
+ });
204
+ // Store instance
205
+ runningServers.set(absolutePath, {
206
+ process: serverProcess,
207
+ socketPath,
208
+ dbPath: absolutePath,
209
+ });
210
+ // Wait for server to be ready
211
+ try {
212
+ await waitForSocket(socketPath, 10000);
213
+ }
214
+ catch (err) {
215
+ // Kill the process if it didn't start properly
216
+ serverProcess.kill();
217
+ runningServers.delete(absolutePath);
218
+ throw new errors_1.DatabaseError(`Failed to start embedded server: ${stderrOutput || err.message}`);
219
+ }
220
+ return socketPath;
221
+ }
222
+ /**
223
+ * Stop the embedded server for a specific database path
224
+ */
225
+ async function stopEmbeddedServer(dbPath) {
226
+ const absolutePath = path.resolve(dbPath);
227
+ const instance = runningServers.get(absolutePath);
228
+ if (!instance) {
229
+ return;
230
+ }
231
+ // Send SIGTERM and wait for graceful shutdown
232
+ instance.process.kill('SIGTERM');
233
+ // Wait for process to exit (max 5 seconds)
234
+ await new Promise((resolve) => {
235
+ const timeout = setTimeout(() => {
236
+ // Force kill if still running
237
+ if (instance.process.exitCode === null) {
238
+ instance.process.kill('SIGKILL');
239
+ }
240
+ resolve();
241
+ }, 5000);
242
+ instance.process.on('exit', () => {
243
+ clearTimeout(timeout);
244
+ resolve();
245
+ });
246
+ });
247
+ runningServers.delete(absolutePath);
248
+ // Clean up socket file
249
+ try {
250
+ if (fs.existsSync(instance.socketPath)) {
251
+ fs.unlinkSync(instance.socketPath);
252
+ }
253
+ }
254
+ catch {
255
+ // Ignore cleanup errors
256
+ }
257
+ }
258
+ /**
259
+ * Stop all running embedded servers
260
+ */
261
+ async function stopAllEmbeddedServers() {
262
+ const stopPromises = [];
263
+ for (const [dbPath] of runningServers) {
264
+ stopPromises.push(stopEmbeddedServer(dbPath));
265
+ }
266
+ await Promise.all(stopPromises);
267
+ }
268
+ /**
269
+ * Check if an embedded server is running for a database path
270
+ */
271
+ function isServerRunning(dbPath) {
272
+ const absolutePath = path.resolve(dbPath);
273
+ const instance = runningServers.get(absolutePath);
274
+ return instance !== undefined && instance.process.exitCode === null;
275
+ }
276
+ // Cleanup on process exit
277
+ process.on('exit', () => {
278
+ for (const instance of runningServers.values()) {
279
+ try {
280
+ instance.process.kill('SIGKILL');
281
+ }
282
+ catch {
283
+ // Ignore
284
+ }
285
+ }
286
+ });
287
+ process.on('SIGINT', async () => {
288
+ await stopAllEmbeddedServers();
289
+ process.exit(0);
290
+ });
291
+ process.on('SIGTERM', async () => {
292
+ await stopAllEmbeddedServers();
293
+ process.exit(0);
294
+ });
295
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server-manager.js","sourceRoot":"","sources":["../../src/server-manager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SochDB Embedded Server Manager\n *\n * Manages the lifecycle of the SochDB server process for embedded mode.\n * Automatically starts the server when needed and stops it on cleanup.\n *\n * @packageDocumentation\n */\n\n// Copyright 2025 Sushanth (https://github.com/sushanthpy)\n//\n// Licensed under the Apache License, Version 2.0 (the \"License\");\n// you may not use this file except in compliance with the License.\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as net from 'net';\nimport { spawn, ChildProcess } from 'child_process';\nimport { ConnectionError, DatabaseError } from './errors';\n\n/**\n * Find the sochdb-server binary (provides IPC interface)\n * \n * Note: sochdb-server uses Unix domain sockets, not available on Windows.\n * On Windows, use the gRPC client instead for cross-platform compatibility.\n */\nfunction findServerBinary(): string {\n  const platform = process.platform;\n  const arch = process.arch;\n\n  // Windows doesn't support Unix sockets, sochdb-server is not available\n  if (platform === 'win32') {\n    throw new DatabaseError(\n      'sochdb-server is not available on Windows (requires Unix domain sockets). ' +\n      'Use the gRPC client for cross-platform support: ' +\n      'const client = await GrpcClient.connect(\"localhost:50051\")'\n    );\n  }\n\n  let target: string;\n  if (platform === 'darwin') {\n    target = arch === 'arm64' ? 'aarch64-apple-darwin' : 'x86_64-apple-darwin';\n  } else {\n    target = arch === 'arm64' ? 'aarch64-unknown-linux-gnu' : 'x86_64-unknown-linux-gnu';\n  }\n\n  const binaryName = 'sochdb-server';\n\n  // Search paths - prioritize bundled binaries\n  const searchPaths = [\n    // Bundled in package (installed via npm) - from dist/cjs or dist/esm\n    path.join(__dirname, '..', '_bin', target, binaryName),\n    path.join(__dirname, '..', '..', '_bin', target, binaryName),\n    path.join(__dirname, '..', '..', '..', '_bin', target, binaryName),\n    // When running from source (src/) during development/testing\n    path.resolve(__dirname, '..', '_bin', target, binaryName),\n    // Development paths - from project root\n    path.join(__dirname, '..', '..', 'target', 'release', binaryName),\n    path.join(__dirname, '..', '..', 'target', 'debug', binaryName),\n    path.join(__dirname, '..', '..', '..', 'target', 'release', binaryName),\n    path.join(__dirname, '..', '..', '..', 'target', 'debug', binaryName),\n    // Absolute paths for sochdb workspace\n    path.resolve(process.cwd(), '_bin', target, binaryName),\n    path.resolve(process.cwd(), 'target', 'release', binaryName),\n    path.resolve(process.cwd(), '..', 'target', 'release', binaryName),\n  ];\n\n  for (const p of searchPaths) {\n    if (fs.existsSync(p)) {\n      return p;\n    }\n  }\n\n  // Try PATH\n  const pathDirs = (process.env.PATH || '').split(path.delimiter);\n  for (const dir of pathDirs) {\n    const p = path.join(dir, binaryName);\n    if (fs.existsSync(p)) {\n      return p;\n    }\n  }\n\n  throw new DatabaseError(\n    `Could not find ${binaryName}. ` +\n    `The pre-built binary may not be available for your platform (${platform}/${arch}). ` +\n    `Install via: cargo build --release -p sochdb-tools`\n  );\n}\n\n/**\n * Wait for a Unix socket to become available\n */\nasync function waitForSocket(socketPath: string, timeoutMs: number = 10000): Promise<void> {\n  const startTime = Date.now();\n  const checkInterval = 100;\n\n  while (Date.now() - startTime < timeoutMs) {\n    if (fs.existsSync(socketPath)) {\n      // Try to connect to verify it's actually listening\n      try {\n        await new Promise<void>((resolve, reject) => {\n          const socket = net.createConnection({ path: socketPath }, () => {\n            socket.destroy();\n            resolve();\n          });\n          socket.on('error', reject);\n          socket.setTimeout(1000);\n        });\n        return;\n      } catch {\n        // Socket exists but not ready yet\n      }\n    }\n    await new Promise(resolve => setTimeout(resolve, checkInterval));\n  }\n\n  throw new ConnectionError(\n    `Timeout waiting for server socket at ${socketPath} after ${timeoutMs}ms`\n  );\n}\n\n/**\n * Embedded server instance\n */\ninterface ServerInstance {\n  process: ChildProcess;\n  socketPath: string;\n  dbPath: string;\n}\n\n// Track running server instances\nconst runningServers: Map<string, ServerInstance> = new Map();\n\n/**\n * Start an embedded SochDB server for the given database path.\n * If a server is already running for this path, return the existing instance.\n *\n * @param dbPath - Path to the database directory\n * @returns The socket path for connecting\n */\nexport async function startEmbeddedServer(dbPath: string): Promise<string> {\n  const absolutePath = path.resolve(dbPath);\n  const socketPath = path.join(absolutePath, 'sochdb.sock');\n\n  // Check if server already running for this path\n  const existing = runningServers.get(absolutePath);\n  if (existing && existing.process.exitCode === null) {\n    // Server still running\n    return socketPath;\n  }\n\n  // Check if another process already has a server running\n  if (fs.existsSync(socketPath)) {\n    try {\n      // Try to connect - if successful, server is running\n      await waitForSocket(socketPath, 1000);\n      return socketPath;\n    } catch {\n      // Socket exists but dead - clean it up\n      try {\n        fs.unlinkSync(socketPath);\n      } catch {\n        // Ignore\n      }\n    }\n  }\n\n  // Ensure database directory exists\n  if (!fs.existsSync(absolutePath)) {\n    fs.mkdirSync(absolutePath, { recursive: true });\n  }\n\n  // Find and spawn server\n  const serverBinary = findServerBinary();\n  \n  const serverProcess = spawn(serverBinary, ['--db', absolutePath, '--socket', socketPath], {\n    stdio: ['ignore', 'pipe', 'pipe'],\n    detached: false,\n  });\n\n  // Collect stderr for error reporting\n  let stderrOutput = '';\n  serverProcess.stderr?.on('data', (data) => {\n    stderrOutput += data.toString();\n  });\n\n  // Handle process exit\n  serverProcess.on('exit', (code, signal) => {\n    runningServers.delete(absolutePath);\n    if (code !== 0 && code !== null) {\n      console.error(`SochDB server exited with code ${code}: ${stderrOutput}`);\n    }\n  });\n\n  serverProcess.on('error', (err) => {\n    runningServers.delete(absolutePath);\n    console.error(`Failed to start SochDB server: ${err.message}`);\n  });\n\n  // Store instance\n  runningServers.set(absolutePath, {\n    process: serverProcess,\n    socketPath,\n    dbPath: absolutePath,\n  });\n\n  // Wait for server to be ready\n  try {\n    await waitForSocket(socketPath, 10000);\n  } catch (err) {\n    // Kill the process if it didn't start properly\n    serverProcess.kill();\n    runningServers.delete(absolutePath);\n    throw new DatabaseError(\n      `Failed to start embedded server: ${stderrOutput || (err as Error).message}`\n    );\n  }\n\n  return socketPath;\n}\n\n/**\n * Stop the embedded server for a specific database path\n */\nexport async function stopEmbeddedServer(dbPath: string): Promise<void> {\n  const absolutePath = path.resolve(dbPath);\n  const instance = runningServers.get(absolutePath);\n\n  if (!instance) {\n    return;\n  }\n\n  // Send SIGTERM and wait for graceful shutdown\n  instance.process.kill('SIGTERM');\n\n  // Wait for process to exit (max 5 seconds)\n  await new Promise<void>((resolve) => {\n    const timeout = setTimeout(() => {\n      // Force kill if still running\n      if (instance.process.exitCode === null) {\n        instance.process.kill('SIGKILL');\n      }\n      resolve();\n    }, 5000);\n\n    instance.process.on('exit', () => {\n      clearTimeout(timeout);\n      resolve();\n    });\n  });\n\n  runningServers.delete(absolutePath);\n\n  // Clean up socket file\n  try {\n    if (fs.existsSync(instance.socketPath)) {\n      fs.unlinkSync(instance.socketPath);\n    }\n  } catch {\n    // Ignore cleanup errors\n  }\n}\n\n/**\n * Stop all running embedded servers\n */\nexport async function stopAllEmbeddedServers(): Promise<void> {\n  const stopPromises: Promise<void>[] = [];\n  for (const [dbPath] of runningServers) {\n    stopPromises.push(stopEmbeddedServer(dbPath));\n  }\n  await Promise.all(stopPromises);\n}\n\n/**\n * Check if an embedded server is running for a database path\n */\nexport function isServerRunning(dbPath: string): boolean {\n  const absolutePath = path.resolve(dbPath);\n  const instance = runningServers.get(absolutePath);\n  return instance !== undefined && instance.process.exitCode === null;\n}\n\n// Cleanup on process exit\nprocess.on('exit', () => {\n  for (const instance of runningServers.values()) {\n    try {\n      instance.process.kill('SIGKILL');\n    } catch {\n      // Ignore\n    }\n  }\n});\n\nprocess.on('SIGINT', async () => {\n  await stopAllEmbeddedServers();\n  process.exit(0);\n});\n\nprocess.on('SIGTERM', async () => {\n  await stopAllEmbeddedServers();\n  process.exit(0);\n});\n"]}