@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.
- package/LICENSE +201 -0
- package/README.md +3349 -0
- package/_bin/aarch64-apple-darwin/libsochdb_storage.dylib +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-bulk +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-grpc-server +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-server +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb-bulk.exe +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb-grpc-server.exe +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb_storage.dll +0 -0
- package/_bin/x86_64-unknown-linux-gnu/libsochdb_storage.so +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-bulk +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-grpc-server +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-server +0 -0
- package/bin/sochdb-bulk.js +80 -0
- package/bin/sochdb-grpc-server.js +80 -0
- package/bin/sochdb-server.js +84 -0
- package/dist/cjs/analytics.js +196 -0
- package/dist/cjs/database.js +929 -0
- package/dist/cjs/embedded/database.js +236 -0
- package/dist/cjs/embedded/ffi/bindings.js +113 -0
- package/dist/cjs/embedded/ffi/library-finder.js +135 -0
- package/dist/cjs/embedded/index.js +14 -0
- package/dist/cjs/embedded/transaction.js +172 -0
- package/dist/cjs/errors.js +71 -0
- package/dist/cjs/format.js +176 -0
- package/dist/cjs/grpc-client.js +328 -0
- package/dist/cjs/index.js +75 -0
- package/dist/cjs/ipc-client.js +504 -0
- package/dist/cjs/query.js +154 -0
- package/dist/cjs/server-manager.js +295 -0
- package/dist/cjs/sql-engine.js +874 -0
- package/dist/esm/analytics.js +196 -0
- package/dist/esm/database.js +931 -0
- package/dist/esm/embedded/database.js +239 -0
- package/dist/esm/embedded/ffi/bindings.js +142 -0
- package/dist/esm/embedded/ffi/library-finder.js +135 -0
- package/dist/esm/embedded/index.js +14 -0
- package/dist/esm/embedded/transaction.js +176 -0
- package/dist/esm/errors.js +71 -0
- package/dist/esm/format.js +179 -0
- package/dist/esm/grpc-client.js +333 -0
- package/dist/esm/index.js +75 -0
- package/dist/esm/ipc-client.js +505 -0
- package/dist/esm/query.js +159 -0
- package/dist/esm/server-manager.js +295 -0
- package/dist/esm/sql-engine.js +875 -0
- package/dist/types/analytics.d.ts +66 -0
- package/dist/types/analytics.d.ts.map +1 -0
- package/dist/types/database.d.ts +523 -0
- package/dist/types/database.d.ts.map +1 -0
- package/dist/types/embedded/database.d.ts +105 -0
- package/dist/types/embedded/database.d.ts.map +1 -0
- package/dist/types/embedded/ffi/bindings.d.ts +24 -0
- package/dist/types/embedded/ffi/bindings.d.ts.map +1 -0
- package/dist/types/embedded/ffi/library-finder.d.ts +17 -0
- package/dist/types/embedded/ffi/library-finder.d.ts.map +1 -0
- package/dist/types/embedded/index.d.ts +9 -0
- package/dist/types/embedded/index.d.ts.map +1 -0
- package/dist/types/embedded/transaction.d.ts +21 -0
- package/dist/types/embedded/transaction.d.ts.map +1 -0
- package/dist/types/errors.d.ts +36 -0
- package/dist/types/errors.d.ts.map +1 -0
- package/dist/types/format.d.ts +117 -0
- package/dist/types/format.d.ts.map +1 -0
- package/dist/types/grpc-client.d.ts +120 -0
- package/dist/types/grpc-client.d.ts.map +1 -0
- package/dist/types/index.d.ts +50 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/ipc-client.d.ts +177 -0
- package/dist/types/ipc-client.d.ts.map +1 -0
- package/dist/types/query.d.ts +85 -0
- package/dist/types/query.d.ts.map +1 -0
- package/dist/types/server-manager.d.ts +29 -0
- package/dist/types/server-manager.d.ts.map +1 -0
- package/dist/types/sql-engine.d.ts +100 -0
- package/dist/types/sql-engine.d.ts.map +1 -0
- package/package.json +90 -0
- package/scripts/postinstall.js +50 -0
|
@@ -0,0 +1,154 @@
|
|
|
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
|
+
constructor(client, pathPrefix) {
|
|
24
|
+
this._client = client;
|
|
25
|
+
this._pathPrefix = pathPrefix;
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Limit the number of results.
|
|
29
|
+
*
|
|
30
|
+
* @param n - Maximum number of results to return
|
|
31
|
+
* @returns This query builder for chaining
|
|
32
|
+
*/
|
|
33
|
+
limit(n) {
|
|
34
|
+
this._limit = n;
|
|
35
|
+
return this;
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Skip the first n results.
|
|
39
|
+
*
|
|
40
|
+
* @param n - Number of results to skip
|
|
41
|
+
* @returns This query builder for chaining
|
|
42
|
+
*/
|
|
43
|
+
offset(n) {
|
|
44
|
+
this._offset = n;
|
|
45
|
+
return this;
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Select specific columns to return.
|
|
49
|
+
*
|
|
50
|
+
* @param columns - Array of column names to select
|
|
51
|
+
* @returns This query builder for chaining
|
|
52
|
+
*/
|
|
53
|
+
select(columns) {
|
|
54
|
+
this._columns = columns;
|
|
55
|
+
return this;
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Execute the query and return results as TOON string.
|
|
59
|
+
*
|
|
60
|
+
* @returns TOON formatted string (e.g., "result[N]{cols}: row1; row2")
|
|
61
|
+
*/
|
|
62
|
+
async execute() {
|
|
63
|
+
return this._client.query(this._pathPrefix, {
|
|
64
|
+
limit: this._limit,
|
|
65
|
+
offset: this._offset,
|
|
66
|
+
columns: this._columns,
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Execute and parse results into a list of objects.
|
|
71
|
+
*
|
|
72
|
+
* @returns Array of result objects
|
|
73
|
+
*/
|
|
74
|
+
async toList() {
|
|
75
|
+
const toonStr = await this.execute();
|
|
76
|
+
return this._parseToon(toonStr);
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Execute and return the first result, or null if none.
|
|
80
|
+
*
|
|
81
|
+
* @returns First result or null
|
|
82
|
+
*/
|
|
83
|
+
async first() {
|
|
84
|
+
const originalLimit = this._limit;
|
|
85
|
+
this._limit = 1;
|
|
86
|
+
const results = await this.toList();
|
|
87
|
+
this._limit = originalLimit;
|
|
88
|
+
return results.length > 0 ? results[0] : null;
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Execute and return the count of results.
|
|
92
|
+
*
|
|
93
|
+
* @returns Number of matching results
|
|
94
|
+
*/
|
|
95
|
+
async count() {
|
|
96
|
+
const results = await this.toList();
|
|
97
|
+
return results.length;
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Simple TOON parser.
|
|
101
|
+
*
|
|
102
|
+
* Parses TOON format: "result[N]{col1,col2}: val1,val2; val3,val4"
|
|
103
|
+
*/
|
|
104
|
+
_parseToon(toonStr) {
|
|
105
|
+
if (!toonStr || toonStr === 'result[0]{}:') {
|
|
106
|
+
return [];
|
|
107
|
+
}
|
|
108
|
+
// Parse header: result[N]{cols}:
|
|
109
|
+
const headerMatch = toonStr.match(/^result\[(\d+)\]\{([^}]*)\}:\s*/);
|
|
110
|
+
if (!headerMatch) {
|
|
111
|
+
// Try to parse as JSON if not TOON format
|
|
112
|
+
try {
|
|
113
|
+
return JSON.parse(toonStr);
|
|
114
|
+
}
|
|
115
|
+
catch {
|
|
116
|
+
return [];
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
const count = parseInt(headerMatch[1], 10);
|
|
120
|
+
if (count === 0) {
|
|
121
|
+
return [];
|
|
122
|
+
}
|
|
123
|
+
const columns = headerMatch[2].split(',').map((c) => c.trim());
|
|
124
|
+
const body = toonStr.substring(headerMatch[0].length);
|
|
125
|
+
// Split rows by semicolon
|
|
126
|
+
const rows = body.split(';').map((r) => r.trim()).filter((r) => r.length > 0);
|
|
127
|
+
return rows.map((row) => {
|
|
128
|
+
const values = row.split(',').map((v) => v.trim());
|
|
129
|
+
const result = {};
|
|
130
|
+
columns.forEach((col, idx) => {
|
|
131
|
+
if (col && idx < values.length) {
|
|
132
|
+
// Try to parse as JSON/number
|
|
133
|
+
let value = values[idx];
|
|
134
|
+
if (value === 'null') {
|
|
135
|
+
value = null;
|
|
136
|
+
}
|
|
137
|
+
else if (value === 'true') {
|
|
138
|
+
value = true;
|
|
139
|
+
}
|
|
140
|
+
else if (value === 'false') {
|
|
141
|
+
value = false;
|
|
142
|
+
}
|
|
143
|
+
else if (!isNaN(Number(value))) {
|
|
144
|
+
value = Number(value);
|
|
145
|
+
}
|
|
146
|
+
result[col] = value;
|
|
147
|
+
}
|
|
148
|
+
});
|
|
149
|
+
return result;
|
|
150
|
+
});
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
exports.Query = Query;
|
|
154
|
+
//# 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"]}
|