@sochdb/sochdb 0.4.2 → 0.4.4
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 +356 -14
- 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/dist/cjs/embedded/database.js +98 -4
- package/dist/cjs/embedded/ffi/bindings.js +46 -8
- package/dist/cjs/index.js +28 -6
- package/dist/cjs/mcp/client.js +115 -0
- package/dist/cjs/mcp/index.js +28 -0
- package/dist/cjs/mcp/server.js +242 -0
- package/dist/cjs/mcp/types.js +32 -0
- package/dist/cjs/namespace.js +147 -19
- package/dist/cjs/policy/index.js +26 -0
- package/dist/cjs/policy/service.js +394 -0
- package/dist/cjs/policy/types.js +8 -0
- package/dist/esm/embedded/database.js +98 -4
- package/dist/esm/embedded/ffi/bindings.js +48 -8
- package/dist/esm/index.js +28 -6
- package/dist/esm/mcp/client.js +116 -0
- package/dist/esm/mcp/index.js +28 -0
- package/dist/esm/mcp/server.js +244 -0
- package/dist/esm/mcp/types.js +34 -0
- package/dist/esm/namespace.js +150 -19
- package/dist/esm/policy/index.js +26 -0
- package/dist/esm/policy/service.js +396 -0
- package/dist/esm/policy/types.js +8 -0
- package/dist/types/embedded/database.d.ts +66 -1
- package/dist/types/embedded/database.d.ts.map +1 -1
- package/dist/types/embedded/ffi/bindings.d.ts +7 -0
- package/dist/types/embedded/ffi/bindings.d.ts.map +1 -1
- package/dist/types/index.d.ts +23 -5
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/mcp/client.d.ts +69 -0
- package/dist/types/mcp/client.d.ts.map +1 -0
- package/dist/types/mcp/index.d.ts +9 -0
- package/dist/types/mcp/index.d.ts.map +1 -0
- package/dist/types/mcp/server.d.ts +87 -0
- package/dist/types/mcp/server.d.ts.map +1 -0
- package/dist/types/mcp/types.d.ts +124 -0
- package/dist/types/mcp/types.d.ts.map +1 -0
- package/dist/types/namespace.d.ts +13 -0
- package/dist/types/namespace.d.ts.map +1 -1
- package/dist/types/policy/index.d.ts +8 -0
- package/dist/types/policy/index.d.ts.map +1 -0
- package/dist/types/policy/service.d.ts +115 -0
- package/dist/types/policy/service.d.ts.map +1 -0
- package/dist/types/policy/types.d.ts +102 -0
- package/dist/types/policy/types.d.ts.map +1 -0
- package/package.json +2 -2
|
@@ -0,0 +1,396 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Policy Service Implementation
|
|
4
|
+
*
|
|
5
|
+
* Policy-based access control and namespace governance for SochDB.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.PolicyService = void 0;
|
|
9
|
+
/**
|
|
10
|
+
* Policy Service for access control and governance
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```typescript
|
|
14
|
+
* import { EmbeddedDatabase, PolicyService } from '@sochdb/sochdb';
|
|
15
|
+
*
|
|
16
|
+
* const db = EmbeddedDatabase.open('./mydb');
|
|
17
|
+
* const policy = new PolicyService(db);
|
|
18
|
+
*
|
|
19
|
+
* // Create a namespace policy
|
|
20
|
+
* await policy.createNamespacePolicy({
|
|
21
|
+
* namespace: 'tenant_123',
|
|
22
|
+
* rules: [{
|
|
23
|
+
* id: 'read_only',
|
|
24
|
+
* name: 'Read Only Access',
|
|
25
|
+
* effect: 'allow',
|
|
26
|
+
* principals: ['user:*'],
|
|
27
|
+
* resources: ['collection:*'],
|
|
28
|
+
* actions: ['read', 'search']
|
|
29
|
+
* }],
|
|
30
|
+
* defaultEffect: 'deny'
|
|
31
|
+
* });
|
|
32
|
+
*
|
|
33
|
+
* // Evaluate access
|
|
34
|
+
* const result = await policy.evaluate({
|
|
35
|
+
* principal: 'user:alice',
|
|
36
|
+
* action: 'read',
|
|
37
|
+
* resource: 'collection:documents'
|
|
38
|
+
* });
|
|
39
|
+
* ```
|
|
40
|
+
*/
|
|
41
|
+
class PolicyService {
|
|
42
|
+
db;
|
|
43
|
+
prefix;
|
|
44
|
+
cache = new Map();
|
|
45
|
+
auditEnabled = true;
|
|
46
|
+
constructor(db, options) {
|
|
47
|
+
this.db = db;
|
|
48
|
+
this.prefix = Buffer.from('_policy:');
|
|
49
|
+
this.auditEnabled = options?.enableAudit ?? true;
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Create a namespace policy
|
|
53
|
+
*/
|
|
54
|
+
async createNamespacePolicy(policy) {
|
|
55
|
+
const key = this.policyKey(policy.namespace);
|
|
56
|
+
const data = {
|
|
57
|
+
...policy,
|
|
58
|
+
createdAt: Date.now(),
|
|
59
|
+
updatedAt: Date.now(),
|
|
60
|
+
};
|
|
61
|
+
await this.db.put(key, Buffer.from(JSON.stringify(data)));
|
|
62
|
+
this.cache.set(policy.namespace, policy);
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Get a namespace policy
|
|
66
|
+
*/
|
|
67
|
+
async getNamespacePolicy(namespace) {
|
|
68
|
+
// Check cache first
|
|
69
|
+
if (this.cache.has(namespace)) {
|
|
70
|
+
return this.cache.get(namespace);
|
|
71
|
+
}
|
|
72
|
+
const key = this.policyKey(namespace);
|
|
73
|
+
const value = await this.db.get(key);
|
|
74
|
+
if (!value) {
|
|
75
|
+
return null;
|
|
76
|
+
}
|
|
77
|
+
const policy = JSON.parse(value.toString());
|
|
78
|
+
this.cache.set(namespace, policy);
|
|
79
|
+
return policy;
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Update a namespace policy
|
|
83
|
+
*/
|
|
84
|
+
async updateNamespacePolicy(namespace, updates) {
|
|
85
|
+
const existing = await this.getNamespacePolicy(namespace);
|
|
86
|
+
if (!existing) {
|
|
87
|
+
throw new Error(`Policy not found for namespace: ${namespace}`);
|
|
88
|
+
}
|
|
89
|
+
const updated = {
|
|
90
|
+
...existing,
|
|
91
|
+
...updates,
|
|
92
|
+
namespace, // Ensure namespace doesn't change
|
|
93
|
+
updatedAt: Date.now(),
|
|
94
|
+
};
|
|
95
|
+
const key = this.policyKey(namespace);
|
|
96
|
+
await this.db.put(key, Buffer.from(JSON.stringify(updated)));
|
|
97
|
+
this.cache.set(namespace, updated);
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Delete a namespace policy
|
|
101
|
+
*/
|
|
102
|
+
async deleteNamespacePolicy(namespace) {
|
|
103
|
+
const key = this.policyKey(namespace);
|
|
104
|
+
await this.db.delete(key);
|
|
105
|
+
this.cache.delete(namespace);
|
|
106
|
+
return true;
|
|
107
|
+
}
|
|
108
|
+
/**
|
|
109
|
+
* Add a rule to a namespace policy
|
|
110
|
+
*/
|
|
111
|
+
async addRule(namespace, rule) {
|
|
112
|
+
const policy = await this.getNamespacePolicy(namespace);
|
|
113
|
+
if (!policy) {
|
|
114
|
+
throw new Error(`Policy not found for namespace: ${namespace}`);
|
|
115
|
+
}
|
|
116
|
+
// Check for duplicate rule ID
|
|
117
|
+
if (policy.rules.some(r => r.id === rule.id)) {
|
|
118
|
+
throw new Error(`Rule with id '${rule.id}' already exists`);
|
|
119
|
+
}
|
|
120
|
+
policy.rules.push(rule);
|
|
121
|
+
await this.updateNamespacePolicy(namespace, { rules: policy.rules });
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Remove a rule from a namespace policy
|
|
125
|
+
*/
|
|
126
|
+
async removeRule(namespace, ruleId) {
|
|
127
|
+
const policy = await this.getNamespacePolicy(namespace);
|
|
128
|
+
if (!policy) {
|
|
129
|
+
return false;
|
|
130
|
+
}
|
|
131
|
+
const index = policy.rules.findIndex(r => r.id === ruleId);
|
|
132
|
+
if (index === -1) {
|
|
133
|
+
return false;
|
|
134
|
+
}
|
|
135
|
+
policy.rules.splice(index, 1);
|
|
136
|
+
await this.updateNamespacePolicy(namespace, { rules: policy.rules });
|
|
137
|
+
return true;
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Evaluate a policy request
|
|
141
|
+
*/
|
|
142
|
+
async evaluate(request) {
|
|
143
|
+
const startTime = Date.now();
|
|
144
|
+
// Extract namespace from resource
|
|
145
|
+
const namespace = this.extractNamespace(request.resource);
|
|
146
|
+
const policy = namespace ? await this.getNamespacePolicy(namespace) : null;
|
|
147
|
+
let result;
|
|
148
|
+
if (!policy) {
|
|
149
|
+
// No policy = allow by default
|
|
150
|
+
result = {
|
|
151
|
+
allowed: true,
|
|
152
|
+
reason: 'No policy defined',
|
|
153
|
+
evaluationTime: Date.now() - startTime,
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
else {
|
|
157
|
+
// Evaluate rules in priority order
|
|
158
|
+
const sortedRules = [...policy.rules].sort((a, b) => (a.priority || 0) - (b.priority || 0));
|
|
159
|
+
let matchedRule;
|
|
160
|
+
for (const rule of sortedRules) {
|
|
161
|
+
if (this.matchesRule(request, rule)) {
|
|
162
|
+
matchedRule = rule;
|
|
163
|
+
break;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
if (matchedRule) {
|
|
167
|
+
result = {
|
|
168
|
+
allowed: matchedRule.effect === 'allow',
|
|
169
|
+
matchedRule,
|
|
170
|
+
reason: `Matched rule: ${matchedRule.name}`,
|
|
171
|
+
evaluationTime: Date.now() - startTime,
|
|
172
|
+
};
|
|
173
|
+
}
|
|
174
|
+
else {
|
|
175
|
+
result = {
|
|
176
|
+
allowed: policy.defaultEffect === 'allow',
|
|
177
|
+
reason: `Default effect: ${policy.defaultEffect}`,
|
|
178
|
+
evaluationTime: Date.now() - startTime,
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
// Log audit entry
|
|
183
|
+
if (this.auditEnabled) {
|
|
184
|
+
await this.logAudit(request, result);
|
|
185
|
+
}
|
|
186
|
+
return result;
|
|
187
|
+
}
|
|
188
|
+
/**
|
|
189
|
+
* Grant namespace access to a principal
|
|
190
|
+
*/
|
|
191
|
+
async grantAccess(grant) {
|
|
192
|
+
const key = this.grantKey(grant.namespace, grant.principal);
|
|
193
|
+
const data = {
|
|
194
|
+
...grant,
|
|
195
|
+
grantedAt: Date.now(),
|
|
196
|
+
};
|
|
197
|
+
await this.db.put(key, Buffer.from(JSON.stringify(data)));
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* Revoke namespace access from a principal
|
|
201
|
+
*/
|
|
202
|
+
async revokeAccess(namespace, principal) {
|
|
203
|
+
const key = this.grantKey(namespace, principal);
|
|
204
|
+
await this.db.delete(key);
|
|
205
|
+
return true;
|
|
206
|
+
}
|
|
207
|
+
/**
|
|
208
|
+
* Check if principal has permission
|
|
209
|
+
*/
|
|
210
|
+
async hasPermission(namespace, principal, permission) {
|
|
211
|
+
const key = this.grantKey(namespace, principal);
|
|
212
|
+
const value = await this.db.get(key);
|
|
213
|
+
if (!value) {
|
|
214
|
+
return false;
|
|
215
|
+
}
|
|
216
|
+
const grant = JSON.parse(value.toString());
|
|
217
|
+
// Check expiration
|
|
218
|
+
if (grant.expiresAt && Date.now() > grant.expiresAt) {
|
|
219
|
+
return false;
|
|
220
|
+
}
|
|
221
|
+
// Admin has all permissions
|
|
222
|
+
if (grant.permissions.includes('admin')) {
|
|
223
|
+
return true;
|
|
224
|
+
}
|
|
225
|
+
return grant.permissions.includes(permission);
|
|
226
|
+
}
|
|
227
|
+
/**
|
|
228
|
+
* List all grants for a namespace
|
|
229
|
+
*/
|
|
230
|
+
async listGrants(namespace) {
|
|
231
|
+
const grants = [];
|
|
232
|
+
const prefix = Buffer.from(`_grant:${namespace}:`);
|
|
233
|
+
try {
|
|
234
|
+
for await (const [_, valueBuffer] of this.db.scanPrefix(prefix)) {
|
|
235
|
+
const grant = JSON.parse(valueBuffer.toString());
|
|
236
|
+
grants.push(grant);
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
catch (error) {
|
|
240
|
+
// Ignore scan errors
|
|
241
|
+
}
|
|
242
|
+
return grants;
|
|
243
|
+
}
|
|
244
|
+
/**
|
|
245
|
+
* Get audit log entries
|
|
246
|
+
*/
|
|
247
|
+
async getAuditLog(options) {
|
|
248
|
+
const entries = [];
|
|
249
|
+
const prefix = Buffer.from('_audit:');
|
|
250
|
+
const limit = options?.limit || 100;
|
|
251
|
+
try {
|
|
252
|
+
for await (const [_, valueBuffer] of this.db.scanPrefix(prefix)) {
|
|
253
|
+
const entry = JSON.parse(valueBuffer.toString());
|
|
254
|
+
// Apply filters
|
|
255
|
+
if (options?.namespace && !entry.resource.includes(options.namespace)) {
|
|
256
|
+
continue;
|
|
257
|
+
}
|
|
258
|
+
if (options?.principal && entry.principal !== options.principal) {
|
|
259
|
+
continue;
|
|
260
|
+
}
|
|
261
|
+
if (options?.action && entry.action !== options.action) {
|
|
262
|
+
continue;
|
|
263
|
+
}
|
|
264
|
+
if (options?.since && entry.timestamp < options.since) {
|
|
265
|
+
continue;
|
|
266
|
+
}
|
|
267
|
+
entries.push(entry);
|
|
268
|
+
if (entries.length >= limit) {
|
|
269
|
+
break;
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
catch (error) {
|
|
274
|
+
// Ignore scan errors
|
|
275
|
+
}
|
|
276
|
+
// Sort by timestamp descending
|
|
277
|
+
return entries.sort((a, b) => b.timestamp - a.timestamp);
|
|
278
|
+
}
|
|
279
|
+
/**
|
|
280
|
+
* Clear policy cache
|
|
281
|
+
*/
|
|
282
|
+
clearCache() {
|
|
283
|
+
this.cache.clear();
|
|
284
|
+
}
|
|
285
|
+
// Private methods
|
|
286
|
+
policyKey(namespace) {
|
|
287
|
+
return Buffer.concat([this.prefix, Buffer.from(`namespace:${namespace}`)]);
|
|
288
|
+
}
|
|
289
|
+
grantKey(namespace, principal) {
|
|
290
|
+
return Buffer.from(`_grant:${namespace}:${principal}`);
|
|
291
|
+
}
|
|
292
|
+
extractNamespace(resource) {
|
|
293
|
+
// Extract namespace from resource like "namespace:tenant_123:collection:docs"
|
|
294
|
+
const parts = resource.split(':');
|
|
295
|
+
if (parts.length >= 2 && parts[0] === 'namespace') {
|
|
296
|
+
return parts[1];
|
|
297
|
+
}
|
|
298
|
+
// Try to extract from collection format "collection:tenant_123:docs"
|
|
299
|
+
if (parts.length >= 2 && parts[0] === 'collection') {
|
|
300
|
+
return parts[1];
|
|
301
|
+
}
|
|
302
|
+
return null;
|
|
303
|
+
}
|
|
304
|
+
matchesRule(request, rule) {
|
|
305
|
+
// Check principals
|
|
306
|
+
if (!this.matchesPatterns(request.principal, rule.principals)) {
|
|
307
|
+
return false;
|
|
308
|
+
}
|
|
309
|
+
// Check resources
|
|
310
|
+
if (!this.matchesPatterns(request.resource, rule.resources)) {
|
|
311
|
+
return false;
|
|
312
|
+
}
|
|
313
|
+
// Check actions
|
|
314
|
+
if (!this.matchesPatterns(request.action, rule.actions)) {
|
|
315
|
+
return false;
|
|
316
|
+
}
|
|
317
|
+
// Check conditions
|
|
318
|
+
if (rule.conditions && rule.conditions.length > 0) {
|
|
319
|
+
for (const condition of rule.conditions) {
|
|
320
|
+
if (!this.evaluateCondition(condition, request.context)) {
|
|
321
|
+
return false;
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
return true;
|
|
326
|
+
}
|
|
327
|
+
matchesPatterns(value, patterns) {
|
|
328
|
+
for (const pattern of patterns) {
|
|
329
|
+
if (this.matchesPattern(value, pattern)) {
|
|
330
|
+
return true;
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
return false;
|
|
334
|
+
}
|
|
335
|
+
matchesPattern(value, pattern) {
|
|
336
|
+
if (pattern === '*') {
|
|
337
|
+
return true;
|
|
338
|
+
}
|
|
339
|
+
if (pattern.endsWith('*')) {
|
|
340
|
+
return value.startsWith(pattern.slice(0, -1));
|
|
341
|
+
}
|
|
342
|
+
if (pattern.startsWith('*')) {
|
|
343
|
+
return value.endsWith(pattern.slice(1));
|
|
344
|
+
}
|
|
345
|
+
return value === pattern;
|
|
346
|
+
}
|
|
347
|
+
evaluateCondition(condition, context) {
|
|
348
|
+
if (!context) {
|
|
349
|
+
return false;
|
|
350
|
+
}
|
|
351
|
+
const value = context[condition.key];
|
|
352
|
+
if (value === undefined) {
|
|
353
|
+
return false;
|
|
354
|
+
}
|
|
355
|
+
switch (condition.operator) {
|
|
356
|
+
case 'equals':
|
|
357
|
+
return value === condition.value;
|
|
358
|
+
case 'not_equals':
|
|
359
|
+
return value !== condition.value;
|
|
360
|
+
case 'contains':
|
|
361
|
+
return String(value).includes(String(condition.value));
|
|
362
|
+
case 'starts_with':
|
|
363
|
+
return String(value).startsWith(String(condition.value));
|
|
364
|
+
case 'ends_with':
|
|
365
|
+
return String(value).endsWith(String(condition.value));
|
|
366
|
+
case 'in':
|
|
367
|
+
return Array.isArray(condition.value) && condition.value.includes(value);
|
|
368
|
+
case 'not_in':
|
|
369
|
+
return Array.isArray(condition.value) && !condition.value.includes(value);
|
|
370
|
+
case 'between':
|
|
371
|
+
if (Array.isArray(condition.value) && condition.value.length === 2) {
|
|
372
|
+
return value >= condition.value[0] && value <= condition.value[1];
|
|
373
|
+
}
|
|
374
|
+
return false;
|
|
375
|
+
default:
|
|
376
|
+
return false;
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
async logAudit(request, result) {
|
|
380
|
+
const entry = {
|
|
381
|
+
id: `${Date.now()}-${Math.random().toString(36).substr(2, 9)}`,
|
|
382
|
+
timestamp: Date.now(),
|
|
383
|
+
principal: request.principal,
|
|
384
|
+
action: request.action,
|
|
385
|
+
resource: request.resource,
|
|
386
|
+
decision: result.allowed ? 'allow' : 'deny',
|
|
387
|
+
matchedRule: result.matchedRule?.id,
|
|
388
|
+
reason: result.reason,
|
|
389
|
+
context: request.context,
|
|
390
|
+
};
|
|
391
|
+
const key = Buffer.from(`_audit:${entry.timestamp}:${entry.id}`);
|
|
392
|
+
await this.db.put(key, Buffer.from(JSON.stringify(entry)));
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
exports.PolicyService = PolicyService;
|
|
396
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service.js","sourceRoot":"","sources":["../../../src/policy/service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Policy Service Implementation\n * \n * Policy-based access control and namespace governance for SochDB.\n */\n\nimport { EmbeddedDatabase } from '../embedded';\nimport {\n  PolicyRule,\n  PolicyCondition,\n  PolicyEvaluation,\n  NamespacePolicy,\n  NamespaceGrant,\n  NamespacePermission,\n  PolicyRequest,\n  PolicySet,\n  PolicyAuditEntry,\n} from './types';\n\n/**\n * Policy Service for access control and governance\n * \n * @example\n * ```typescript\n * import { EmbeddedDatabase, PolicyService } from '@sochdb/sochdb';\n * \n * const db = EmbeddedDatabase.open('./mydb');\n * const policy = new PolicyService(db);\n * \n * // Create a namespace policy\n * await policy.createNamespacePolicy({\n *   namespace: 'tenant_123',\n *   rules: [{\n *     id: 'read_only',\n *     name: 'Read Only Access',\n *     effect: 'allow',\n *     principals: ['user:*'],\n *     resources: ['collection:*'],\n *     actions: ['read', 'search']\n *   }],\n *   defaultEffect: 'deny'\n * });\n * \n * // Evaluate access\n * const result = await policy.evaluate({\n *   principal: 'user:alice',\n *   action: 'read',\n *   resource: 'collection:documents'\n * });\n * ```\n */\nexport class PolicyService {\n  private db: EmbeddedDatabase;\n  private prefix: Buffer;\n  private cache: Map<string, NamespacePolicy> = new Map();\n  private auditEnabled = true;\n\n  constructor(db: EmbeddedDatabase, options?: { enableAudit?: boolean }) {\n    this.db = db;\n    this.prefix = Buffer.from('_policy:');\n    this.auditEnabled = options?.enableAudit ?? true;\n  }\n\n  /**\n   * Create a namespace policy\n   */\n  async createNamespacePolicy(policy: NamespacePolicy): Promise<void> {\n    const key = this.policyKey(policy.namespace);\n    const data = {\n      ...policy,\n      createdAt: Date.now(),\n      updatedAt: Date.now(),\n    };\n    \n    await this.db.put(key, Buffer.from(JSON.stringify(data)));\n    this.cache.set(policy.namespace, policy);\n  }\n\n  /**\n   * Get a namespace policy\n   */\n  async getNamespacePolicy(namespace: string): Promise<NamespacePolicy | null> {\n    // Check cache first\n    if (this.cache.has(namespace)) {\n      return this.cache.get(namespace)!;\n    }\n\n    const key = this.policyKey(namespace);\n    const value = await this.db.get(key);\n    \n    if (!value) {\n      return null;\n    }\n\n    const policy = JSON.parse(value.toString()) as NamespacePolicy;\n    this.cache.set(namespace, policy);\n    return policy;\n  }\n\n  /**\n   * Update a namespace policy\n   */\n  async updateNamespacePolicy(namespace: string, updates: Partial<NamespacePolicy>): Promise<void> {\n    const existing = await this.getNamespacePolicy(namespace);\n    if (!existing) {\n      throw new Error(`Policy not found for namespace: ${namespace}`);\n    }\n\n    const updated = {\n      ...existing,\n      ...updates,\n      namespace, // Ensure namespace doesn't change\n      updatedAt: Date.now(),\n    };\n\n    const key = this.policyKey(namespace);\n    await this.db.put(key, Buffer.from(JSON.stringify(updated)));\n    this.cache.set(namespace, updated);\n  }\n\n  /**\n   * Delete a namespace policy\n   */\n  async deleteNamespacePolicy(namespace: string): Promise<boolean> {\n    const key = this.policyKey(namespace);\n    await this.db.delete(key);\n    this.cache.delete(namespace);\n    return true;\n  }\n\n  /**\n   * Add a rule to a namespace policy\n   */\n  async addRule(namespace: string, rule: PolicyRule): Promise<void> {\n    const policy = await this.getNamespacePolicy(namespace);\n    if (!policy) {\n      throw new Error(`Policy not found for namespace: ${namespace}`);\n    }\n\n    // Check for duplicate rule ID\n    if (policy.rules.some(r => r.id === rule.id)) {\n      throw new Error(`Rule with id '${rule.id}' already exists`);\n    }\n\n    policy.rules.push(rule);\n    await this.updateNamespacePolicy(namespace, { rules: policy.rules });\n  }\n\n  /**\n   * Remove a rule from a namespace policy\n   */\n  async removeRule(namespace: string, ruleId: string): Promise<boolean> {\n    const policy = await this.getNamespacePolicy(namespace);\n    if (!policy) {\n      return false;\n    }\n\n    const index = policy.rules.findIndex(r => r.id === ruleId);\n    if (index === -1) {\n      return false;\n    }\n\n    policy.rules.splice(index, 1);\n    await this.updateNamespacePolicy(namespace, { rules: policy.rules });\n    return true;\n  }\n\n  /**\n   * Evaluate a policy request\n   */\n  async evaluate(request: PolicyRequest): Promise<PolicyEvaluation> {\n    const startTime = Date.now();\n    \n    // Extract namespace from resource\n    const namespace = this.extractNamespace(request.resource);\n    const policy = namespace ? await this.getNamespacePolicy(namespace) : null;\n\n    let result: PolicyEvaluation;\n\n    if (!policy) {\n      // No policy = allow by default\n      result = {\n        allowed: true,\n        reason: 'No policy defined',\n        evaluationTime: Date.now() - startTime,\n      };\n    } else {\n      // Evaluate rules in priority order\n      const sortedRules = [...policy.rules].sort((a, b) => \n        (a.priority || 0) - (b.priority || 0)\n      );\n\n      let matchedRule: PolicyRule | undefined;\n\n      for (const rule of sortedRules) {\n        if (this.matchesRule(request, rule)) {\n          matchedRule = rule;\n          break;\n        }\n      }\n\n      if (matchedRule) {\n        result = {\n          allowed: matchedRule.effect === 'allow',\n          matchedRule,\n          reason: `Matched rule: ${matchedRule.name}`,\n          evaluationTime: Date.now() - startTime,\n        };\n      } else {\n        result = {\n          allowed: policy.defaultEffect === 'allow',\n          reason: `Default effect: ${policy.defaultEffect}`,\n          evaluationTime: Date.now() - startTime,\n        };\n      }\n    }\n\n    // Log audit entry\n    if (this.auditEnabled) {\n      await this.logAudit(request, result);\n    }\n\n    return result;\n  }\n\n  /**\n   * Grant namespace access to a principal\n   */\n  async grantAccess(grant: NamespaceGrant): Promise<void> {\n    const key = this.grantKey(grant.namespace, grant.principal);\n    const data = {\n      ...grant,\n      grantedAt: Date.now(),\n    };\n    \n    await this.db.put(key, Buffer.from(JSON.stringify(data)));\n  }\n\n  /**\n   * Revoke namespace access from a principal\n   */\n  async revokeAccess(namespace: string, principal: string): Promise<boolean> {\n    const key = this.grantKey(namespace, principal);\n    await this.db.delete(key);\n    return true;\n  }\n\n  /**\n   * Check if principal has permission\n   */\n  async hasPermission(\n    namespace: string,\n    principal: string,\n    permission: NamespacePermission\n  ): Promise<boolean> {\n    const key = this.grantKey(namespace, principal);\n    const value = await this.db.get(key);\n    \n    if (!value) {\n      return false;\n    }\n\n    const grant = JSON.parse(value.toString()) as NamespaceGrant;\n\n    // Check expiration\n    if (grant.expiresAt && Date.now() > grant.expiresAt) {\n      return false;\n    }\n\n    // Admin has all permissions\n    if (grant.permissions.includes('admin')) {\n      return true;\n    }\n\n    return grant.permissions.includes(permission);\n  }\n\n  /**\n   * List all grants for a namespace\n   */\n  async listGrants(namespace: string): Promise<NamespaceGrant[]> {\n    const grants: NamespaceGrant[] = [];\n    const prefix = Buffer.from(`_grant:${namespace}:`);\n\n    try {\n      for await (const [_, valueBuffer] of this.db.scanPrefix(prefix)) {\n        const grant = JSON.parse(valueBuffer.toString()) as NamespaceGrant;\n        grants.push(grant);\n      }\n    } catch (error) {\n      // Ignore scan errors\n    }\n\n    return grants;\n  }\n\n  /**\n   * Get audit log entries\n   */\n  async getAuditLog(options?: {\n    namespace?: string;\n    principal?: string;\n    action?: string;\n    since?: number;\n    limit?: number;\n  }): Promise<PolicyAuditEntry[]> {\n    const entries: PolicyAuditEntry[] = [];\n    const prefix = Buffer.from('_audit:');\n    const limit = options?.limit || 100;\n\n    try {\n      for await (const [_, valueBuffer] of this.db.scanPrefix(prefix)) {\n        const entry = JSON.parse(valueBuffer.toString()) as PolicyAuditEntry;\n        \n        // Apply filters\n        if (options?.namespace && !entry.resource.includes(options.namespace)) {\n          continue;\n        }\n        if (options?.principal && entry.principal !== options.principal) {\n          continue;\n        }\n        if (options?.action && entry.action !== options.action) {\n          continue;\n        }\n        if (options?.since && entry.timestamp < options.since) {\n          continue;\n        }\n\n        entries.push(entry);\n        \n        if (entries.length >= limit) {\n          break;\n        }\n      }\n    } catch (error) {\n      // Ignore scan errors\n    }\n\n    // Sort by timestamp descending\n    return entries.sort((a, b) => b.timestamp - a.timestamp);\n  }\n\n  /**\n   * Clear policy cache\n   */\n  clearCache(): void {\n    this.cache.clear();\n  }\n\n  // Private methods\n\n  private policyKey(namespace: string): Buffer {\n    return Buffer.concat([this.prefix, Buffer.from(`namespace:${namespace}`)]);\n  }\n\n  private grantKey(namespace: string, principal: string): Buffer {\n    return Buffer.from(`_grant:${namespace}:${principal}`);\n  }\n\n  private extractNamespace(resource: string): string | null {\n    // Extract namespace from resource like \"namespace:tenant_123:collection:docs\"\n    const parts = resource.split(':');\n    if (parts.length >= 2 && parts[0] === 'namespace') {\n      return parts[1];\n    }\n    // Try to extract from collection format \"collection:tenant_123:docs\"\n    if (parts.length >= 2 && parts[0] === 'collection') {\n      return parts[1];\n    }\n    return null;\n  }\n\n  private matchesRule(request: PolicyRequest, rule: PolicyRule): boolean {\n    // Check principals\n    if (!this.matchesPatterns(request.principal, rule.principals)) {\n      return false;\n    }\n\n    // Check resources\n    if (!this.matchesPatterns(request.resource, rule.resources)) {\n      return false;\n    }\n\n    // Check actions\n    if (!this.matchesPatterns(request.action, rule.actions)) {\n      return false;\n    }\n\n    // Check conditions\n    if (rule.conditions && rule.conditions.length > 0) {\n      for (const condition of rule.conditions) {\n        if (!this.evaluateCondition(condition, request.context)) {\n          return false;\n        }\n      }\n    }\n\n    return true;\n  }\n\n  private matchesPatterns(value: string, patterns: string[]): boolean {\n    for (const pattern of patterns) {\n      if (this.matchesPattern(value, pattern)) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  private matchesPattern(value: string, pattern: string): boolean {\n    if (pattern === '*') {\n      return true;\n    }\n    \n    if (pattern.endsWith('*')) {\n      return value.startsWith(pattern.slice(0, -1));\n    }\n    \n    if (pattern.startsWith('*')) {\n      return value.endsWith(pattern.slice(1));\n    }\n    \n    return value === pattern;\n  }\n\n  private evaluateCondition(condition: PolicyCondition, context?: Record<string, any>): boolean {\n    if (!context) {\n      return false;\n    }\n\n    const value = context[condition.key];\n    if (value === undefined) {\n      return false;\n    }\n\n    switch (condition.operator) {\n      case 'equals':\n        return value === condition.value;\n      case 'not_equals':\n        return value !== condition.value;\n      case 'contains':\n        return String(value).includes(String(condition.value));\n      case 'starts_with':\n        return String(value).startsWith(String(condition.value));\n      case 'ends_with':\n        return String(value).endsWith(String(condition.value));\n      case 'in':\n        return Array.isArray(condition.value) && condition.value.includes(value);\n      case 'not_in':\n        return Array.isArray(condition.value) && !condition.value.includes(value);\n      case 'between':\n        if (Array.isArray(condition.value) && condition.value.length === 2) {\n          return value >= condition.value[0] && value <= condition.value[1];\n        }\n        return false;\n      default:\n        return false;\n    }\n  }\n\n  private async logAudit(request: PolicyRequest, result: PolicyEvaluation): Promise<void> {\n    const entry: PolicyAuditEntry = {\n      id: `${Date.now()}-${Math.random().toString(36).substr(2, 9)}`,\n      timestamp: Date.now(),\n      principal: request.principal,\n      action: request.action,\n      resource: request.resource,\n      decision: result.allowed ? 'allow' : 'deny',\n      matchedRule: result.matchedRule?.id,\n      reason: result.reason,\n      context: request.context,\n    };\n\n    const key = Buffer.from(`_audit:${entry.timestamp}:${entry.id}`);\n    await this.db.put(key, Buffer.from(JSON.stringify(entry)));\n  }\n}\n"]}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Policy Service Types
|
|
4
|
+
*
|
|
5
|
+
* Type definitions for policy-based access control and namespace governance.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvcG9saWN5L3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7OztHQUlHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBQb2xpY3kgU2VydmljZSBUeXBlc1xuICogXG4gKiBUeXBlIGRlZmluaXRpb25zIGZvciBwb2xpY3ktYmFzZWQgYWNjZXNzIGNvbnRyb2wgYW5kIG5hbWVzcGFjZSBnb3Zlcm5hbmNlLlxuICovXG5cbi8qKlxuICogUG9saWN5IHJ1bGUgZm9yIGFjY2VzcyBjb250cm9sXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUG9saWN5UnVsZSB7XG4gIGlkOiBzdHJpbmc7XG4gIG5hbWU6IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIGVmZmVjdDogJ2FsbG93JyB8ICdkZW55JztcbiAgcHJpbmNpcGFsczogc3RyaW5nW107XG4gIHJlc291cmNlczogc3RyaW5nW107XG4gIGFjdGlvbnM6IHN0cmluZ1tdO1xuICBjb25kaXRpb25zPzogUG9saWN5Q29uZGl0aW9uW107XG4gIHByaW9yaXR5PzogbnVtYmVyO1xufVxuXG4vKipcbiAqIFBvbGljeSBjb25kaXRpb24gZm9yIGNvbnRleHR1YWwgYWNjZXNzIGNvbnRyb2xcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBQb2xpY3lDb25kaXRpb24ge1xuICB0eXBlOiAndGltZScgfCAnaXAnIHwgJ21ldGFkYXRhJyB8ICdjdXN0b20nO1xuICBvcGVyYXRvcjogJ2VxdWFscycgfCAnbm90X2VxdWFscycgfCAnY29udGFpbnMnIHwgJ3N0YXJ0c193aXRoJyB8ICdlbmRzX3dpdGgnIHwgJ2luJyB8ICdub3RfaW4nIHwgJ2JldHdlZW4nO1xuICBrZXk6IHN0cmluZztcbiAgdmFsdWU6IGFueTtcbn1cblxuLyoqXG4gKiBQb2xpY3kgZXZhbHVhdGlvbiByZXN1bHRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBQb2xpY3lFdmFsdWF0aW9uIHtcbiAgYWxsb3dlZDogYm9vbGVhbjtcbiAgbWF0Y2hlZFJ1bGU/OiBQb2xpY3lSdWxlO1xuICByZWFzb24/OiBzdHJpbmc7XG4gIGV2YWx1YXRpb25UaW1lOiBudW1iZXI7XG59XG5cbi8qKlxuICogTmFtZXNwYWNlIHBvbGljeSBmb3IgbXVsdGktdGVuYW50IGlzb2xhdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIE5hbWVzcGFjZVBvbGljeSB7XG4gIG5hbWVzcGFjZTogc3RyaW5nO1xuICBydWxlczogUG9saWN5UnVsZVtdO1xuICBkZWZhdWx0RWZmZWN0OiAnYWxsb3cnIHwgJ2RlbnknO1xuICBpbmhlcml0RnJvbT86IHN0cmluZztcbiAgbWV0YWRhdGE/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xufVxuXG4vKipcbiAqIEFjY2VzcyBncmFudCBmb3IgbmFtZXNwYWNlLWxldmVsIHBlcm1pc3Npb25zXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTmFtZXNwYWNlR3JhbnQge1xuICBuYW1lc3BhY2U6IHN0cmluZztcbiAgcHJpbmNpcGFsOiBzdHJpbmc7XG4gIHBlcm1pc3Npb25zOiBOYW1lc3BhY2VQZXJtaXNzaW9uW107XG4gIGV4cGlyZXNBdD86IG51bWJlcjtcbiAgZ3JhbnRlZEJ5Pzogc3RyaW5nO1xuICBncmFudGVkQXQ/OiBudW1iZXI7XG59XG5cbi8qKlxuICogTmFtZXNwYWNlIHBlcm1pc3Npb24gbGV2ZWxzXG4gKi9cbmV4cG9ydCB0eXBlIE5hbWVzcGFjZVBlcm1pc3Npb24gPSBcbiAgfCAncmVhZCdcbiAgfCAnd3JpdGUnXG4gIHwgJ2RlbGV0ZSdcbiAgfCAnYWRtaW4nXG4gIHwgJ2NyZWF0ZV9jb2xsZWN0aW9uJ1xuICB8ICdkZWxldGVfY29sbGVjdGlvbidcbiAgfCAnc2VhcmNoJ1xuICB8ICdtYW5hZ2VfcG9saWN5JztcblxuLyoqXG4gKiBQb2xpY3kgYWN0aW9uIHR5cGVzXG4gKi9cbmV4cG9ydCB0eXBlIFBvbGljeUFjdGlvbiA9XG4gIHwgJ2RiOnJlYWQnXG4gIHwgJ2RiOndyaXRlJ1xuICB8ICdkYjpkZWxldGUnXG4gIHwgJ2RiOnNjYW4nXG4gIHwgJ25hbWVzcGFjZTpjcmVhdGUnXG4gIHwgJ25hbWVzcGFjZTpkZWxldGUnXG4gIHwgJ25hbWVzcGFjZTpsaXN0J1xuICB8ICdjb2xsZWN0aW9uOmNyZWF0ZSdcbiAgfCAnY29sbGVjdGlvbjpkZWxldGUnXG4gIHwgJ2NvbGxlY3Rpb246aW5zZXJ0J1xuICB8ICdjb2xsZWN0aW9uOnNlYXJjaCdcbiAgfCAnY29sbGVjdGlvbjp1cGRhdGUnXG4gIHwgJ3BvbGljeTpyZWFkJ1xuICB8ICdwb2xpY3k6d3JpdGUnXG4gIHwgJ2FkbWluOionO1xuXG4vKipcbiAqIFBvbGljeSByZXF1ZXN0IGZvciBldmFsdWF0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUG9saWN5UmVxdWVzdCB7XG4gIHByaW5jaXBhbDogc3RyaW5nO1xuICBhY3Rpb246IHN0cmluZztcbiAgcmVzb3VyY2U6IHN0cmluZztcbiAgY29udGV4dD86IFJlY29yZDxzdHJpbmcsIGFueT47XG59XG5cbi8qKlxuICogUG9saWN5IHNldCBjb250YWluaW5nIG11bHRpcGxlIHBvbGljaWVzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUG9saWN5U2V0IHtcbiAgaWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgcG9saWNpZXM6IE5hbWVzcGFjZVBvbGljeVtdO1xuICB2ZXJzaW9uOiBudW1iZXI7XG4gIGNyZWF0ZWRBdDogbnVtYmVyO1xuICB1cGRhdGVkQXQ6IG51bWJlcjtcbn1cblxuLyoqXG4gKiBBdWRpdCBsb2cgZW50cnkgZm9yIHBvbGljeSBkZWNpc2lvbnNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBQb2xpY3lBdWRpdEVudHJ5IHtcbiAgaWQ6IHN0cmluZztcbiAgdGltZXN0YW1wOiBudW1iZXI7XG4gIHByaW5jaXBhbDogc3RyaW5nO1xuICBhY3Rpb246IHN0cmluZztcbiAgcmVzb3VyY2U6IHN0cmluZztcbiAgZGVjaXNpb246ICdhbGxvdycgfCAnZGVueSc7XG4gIG1hdGNoZWRSdWxlPzogc3RyaW5nO1xuICByZWFzb24/OiBzdHJpbmc7XG4gIGNvbnRleHQ/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xufVxuIl19
|
|
@@ -31,15 +31,80 @@ export declare class EmbeddedDatabase {
|
|
|
31
31
|
private bindings;
|
|
32
32
|
private closed;
|
|
33
33
|
private path;
|
|
34
|
+
private concurrent;
|
|
35
|
+
private _concurrentModeFallback;
|
|
34
36
|
private constructor();
|
|
35
37
|
/**
|
|
36
|
-
* Open a database at the specified path
|
|
38
|
+
* Open a database at the specified path in standard mode
|
|
39
|
+
*
|
|
40
|
+
* For web applications with multiple processes, use `openConcurrent()` instead.
|
|
37
41
|
*
|
|
38
42
|
* @param path - Path to database directory
|
|
39
43
|
* @param config - Optional configuration
|
|
40
44
|
* @returns EmbeddedDatabase instance
|
|
41
45
|
*/
|
|
42
46
|
static open(path: string, config?: EmbeddedDatabaseConfig): EmbeddedDatabase;
|
|
47
|
+
/**
|
|
48
|
+
* Open a database in concurrent mode for multi-process web applications
|
|
49
|
+
*
|
|
50
|
+
* This mode allows multiple Node.js processes (e.g., PM2 cluster workers,
|
|
51
|
+
* multiple Express instances) to access the database simultaneously.
|
|
52
|
+
*
|
|
53
|
+
* Features:
|
|
54
|
+
* - Lock-free reads with ~100ns latency
|
|
55
|
+
* - Multi-reader, single-writer coordination
|
|
56
|
+
* - Automatic write serialization
|
|
57
|
+
*
|
|
58
|
+
* @example
|
|
59
|
+
* ```typescript
|
|
60
|
+
* import { EmbeddedDatabase } from '@sochdb/sochdb';
|
|
61
|
+
* import express from 'express';
|
|
62
|
+
*
|
|
63
|
+
* // Open in concurrent mode - multiple workers can access
|
|
64
|
+
* const db = EmbeddedDatabase.openConcurrent('./web_db');
|
|
65
|
+
*
|
|
66
|
+
* const app = express();
|
|
67
|
+
*
|
|
68
|
+
* app.get('/user/:id', async (req, res) => {
|
|
69
|
+
* // Multiple concurrent requests can read simultaneously
|
|
70
|
+
* const data = await db.get(Buffer.from(`user:${req.params.id}`));
|
|
71
|
+
* if (!data) {
|
|
72
|
+
* res.status(404).json({ error: 'not found' });
|
|
73
|
+
* return;
|
|
74
|
+
* }
|
|
75
|
+
* res.send(data);
|
|
76
|
+
* });
|
|
77
|
+
*
|
|
78
|
+
* app.post('/user/:id', async (req, res) => {
|
|
79
|
+
* // Writes are serialized automatically
|
|
80
|
+
* await db.put(Buffer.from(`user:${req.params.id}`), req.body);
|
|
81
|
+
* res.json({ status: 'ok' });
|
|
82
|
+
* });
|
|
83
|
+
*
|
|
84
|
+
* // Start with PM2 cluster mode:
|
|
85
|
+
* // pm2 start app.js -i max
|
|
86
|
+
* app.listen(3000);
|
|
87
|
+
* ```
|
|
88
|
+
*
|
|
89
|
+
* @param path - Path to database directory
|
|
90
|
+
* @param options - Optional configuration for concurrent mode
|
|
91
|
+
* @returns EmbeddedDatabase instance in concurrent mode
|
|
92
|
+
*/
|
|
93
|
+
static openConcurrent(path: string, options?: {
|
|
94
|
+
fallbackToStandard?: boolean;
|
|
95
|
+
}): EmbeddedDatabase;
|
|
96
|
+
/**
|
|
97
|
+
* Check if database is opened in concurrent mode
|
|
98
|
+
*/
|
|
99
|
+
get isConcurrent(): boolean;
|
|
100
|
+
/**
|
|
101
|
+
* Check if concurrent mode fell back to standard mode
|
|
102
|
+
*/
|
|
103
|
+
get isConcurrentFallback(): boolean;
|
|
104
|
+
/**
|
|
105
|
+
* Check if concurrent mode is available in the native library
|
|
106
|
+
*/
|
|
107
|
+
static isConcurrentModeAvailable(): boolean;
|
|
43
108
|
/**
|
|
44
109
|
* Put a key-value pair (auto-transaction)
|
|
45
110
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"database.d.ts","sourceRoot":"","sources":["../../../src/embedded/database.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAChD,OAAO,EAAE,mBAAmB,EAAE,MAAM,eAAe,CAAC;AAGpD,MAAM,WAAW,sBAAsB;IACnC,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,QAAQ,CAAC,EAAE,MAAM,GAAG,QAAQ,GAAG,KAAK,CAAC;IACrC,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,WAAW,CAAC,EAAE,iBAAiB,GAAG,UAAU,GAAG,gBAAgB,GAAG,aAAa,CAAC;CACnF;AAED;;;;;;;;;;;;GAYG;AACH,qBAAa,gBAAgB;IACzB,OAAO,CAAC,MAAM,CAAM;IACpB,OAAO,CAAC,QAAQ,CAAiB;IACjC,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,IAAI,CAAS;
|
|
1
|
+
{"version":3,"file":"database.d.ts","sourceRoot":"","sources":["../../../src/embedded/database.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAChD,OAAO,EAAE,mBAAmB,EAAE,MAAM,eAAe,CAAC;AAGpD,MAAM,WAAW,sBAAsB;IACnC,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,QAAQ,CAAC,EAAE,MAAM,GAAG,QAAQ,GAAG,KAAK,CAAC;IACrC,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,WAAW,CAAC,EAAE,iBAAiB,GAAG,UAAU,GAAG,gBAAgB,GAAG,aAAa,CAAC;CACnF;AAED;;;;;;;;;;;;GAYG;AACH,qBAAa,gBAAgB;IACzB,OAAO,CAAC,MAAM,CAAM;IACpB,OAAO,CAAC,QAAQ,CAAiB;IACjC,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,IAAI,CAAS;IACrB,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,uBAAuB,CAAS;IAExC,OAAO;IAQP;;;;;;;;OAQG;IACH,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,sBAAsB,GAAG,gBAAgB;IA4B5E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA6CG;IACH,MAAM,CAAC,cAAc,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE;QAAE,kBAAkB,CAAC,EAAE,OAAO,CAAA;KAAE,GAAG,gBAAgB;IAgCjG;;OAEG;IACH,IAAI,YAAY,IAAI,OAAO,CAE1B;IAED;;OAEG;IACH,IAAI,oBAAoB,IAAI,OAAO,CAElC;IAED;;OAEG;IACH,MAAM,CAAC,yBAAyB,IAAI,OAAO;IAI3C;;OAEG;IACG,GAAG,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAapD;;OAEG;IACG,GAAG,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;IAc9C;;OAEG;IACG,MAAM,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAaxC;;OAEG;IACG,OAAO,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAazD;;OAEG;IACG,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;IAcnD;;OAEG;IACI,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,cAAc,CAAC,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAenE;;OAEG;IACH,WAAW,IAAI,mBAAmB;IAOlC;;OAEG;IACG,eAAe,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,EAAE,mBAAmB,KAAK,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAYlF;;OAEG;IACG,UAAU,IAAI,OAAO,CAAC,MAAM,CAAC;IAMnC;;OAEG;IACG,KAAK,IAAI,OAAO,CAAC;QACnB,iBAAiB,EAAE,MAAM,CAAC;QAC1B,YAAY,EAAE,MAAM,CAAC;QACrB,kBAAkB,EAAE,MAAM,CAAC;QAC3B,iBAAiB,EAAE,MAAM,CAAC;QAC1B,iBAAiB,EAAE,MAAM,CAAC;KAC7B,CAAC;IAiBF;;OAEG;IACH,KAAK,IAAI,IAAI;IAOb,OAAO,CAAC,UAAU;IAMlB;;;OAGG;IACH,SAAS,IAAI,GAAG;IAIhB;;;OAGG;IACH,WAAW,IAAI,cAAc;CAGhC"}
|
|
@@ -1,8 +1,11 @@
|
|
|
1
1
|
export declare class NativeBindings {
|
|
2
2
|
private static instance;
|
|
3
3
|
private lib;
|
|
4
|
+
private _concurrentModeAvailable;
|
|
4
5
|
sochdb_open: any;
|
|
5
6
|
sochdb_open_with_config: any;
|
|
7
|
+
sochdb_open_concurrent: any;
|
|
8
|
+
sochdb_is_concurrent: any;
|
|
6
9
|
sochdb_close: any;
|
|
7
10
|
sochdb_begin_txn: any;
|
|
8
11
|
sochdb_commit: any;
|
|
@@ -20,5 +23,9 @@ export declare class NativeBindings {
|
|
|
20
23
|
sochdb_free_bytes: any;
|
|
21
24
|
private constructor();
|
|
22
25
|
static getInstance(): NativeBindings;
|
|
26
|
+
/**
|
|
27
|
+
* Check if concurrent mode is available in the native library
|
|
28
|
+
*/
|
|
29
|
+
isConcurrentModeAvailable(): boolean;
|
|
23
30
|
}
|
|
24
31
|
//# sourceMappingURL=bindings.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"bindings.d.ts","sourceRoot":"","sources":["../../../../src/embedded/ffi/bindings.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"bindings.d.ts","sourceRoot":"","sources":["../../../../src/embedded/ffi/bindings.ts"],"names":[],"mappings":"AAyDA,qBAAa,cAAc;IACvB,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAiB;IACxC,OAAO,CAAC,GAAG,CAAM;IACjB,OAAO,CAAC,wBAAwB,CAAS;IAGlC,WAAW,EAAE,GAAG,CAAC;IACjB,uBAAuB,EAAE,GAAG,CAAC;IAC7B,sBAAsB,EAAE,GAAG,CAAC;IAC5B,oBAAoB,EAAE,GAAG,CAAC;IAC1B,YAAY,EAAE,GAAG,CAAC;IAGlB,gBAAgB,EAAE,GAAG,CAAC;IACtB,aAAa,EAAE,GAAG,CAAC;IACnB,YAAY,EAAE,GAAG,CAAC;IAIlB,UAAU,EAAE,GAAG,CAAC;IAEhB,UAAU,EAAE,GAAG,CAAC;IAEhB,aAAa,EAAE,GAAG,CAAC;IAGnB,eAAe,EAAE,GAAG,CAAC;IACrB,eAAe,EAAE,GAAG,CAAC;IAGrB,kBAAkB,EAAE,GAAG,CAAC;IACxB,oBAAoB,EAAE,GAAG,CAAC;IAC1B,qBAAqB,EAAE,GAAG,CAAC;IAG3B,YAAY,EAAE,GAAG,CAAC;IAClB,iBAAiB,EAAE,GAAG,CAAC;IAGvB,iBAAiB,EAAE,GAAG,CAAC;IAE9B,OAAO;WAwDO,WAAW,IAAI,cAAc;IAO3C;;OAEG;IACI,yBAAyB,IAAI,OAAO;CAG9C"}
|
package/dist/types/index.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* SochDB Node.js SDK v0.4.
|
|
2
|
+
* SochDB Node.js SDK v0.4.4
|
|
3
3
|
*
|
|
4
4
|
* Dual-mode architecture: Embedded (FFI) + Server (gRPC/IPC)
|
|
5
5
|
*
|
|
@@ -12,7 +12,12 @@
|
|
|
12
12
|
* - No server required - just npm install and run
|
|
13
13
|
* - Best for: Local development, simple apps
|
|
14
14
|
*
|
|
15
|
-
* 2.
|
|
15
|
+
* 2. Concurrent Mode (FFI) - For multi-process apps:
|
|
16
|
+
* - Same FFI bindings with MVCC for concurrent access
|
|
17
|
+
* - Multiple Node.js processes can access same database
|
|
18
|
+
* - Best for: PM2 cluster, Express workers
|
|
19
|
+
*
|
|
20
|
+
* 3. Server Mode (gRPC/IPC) - For distributed systems:
|
|
16
21
|
* - Thin client connecting to sochdb-grpc server
|
|
17
22
|
* - Best for: Production, multi-language, scalability
|
|
18
23
|
*
|
|
@@ -26,6 +31,15 @@
|
|
|
26
31
|
* await db.close();
|
|
27
32
|
* ```
|
|
28
33
|
*
|
|
34
|
+
* @example Concurrent Mode
|
|
35
|
+
* ```typescript
|
|
36
|
+
* import { Database } from '@sochdb/sochdb';
|
|
37
|
+
*
|
|
38
|
+
* // Multiple processes can access simultaneously
|
|
39
|
+
* const db = Database.openConcurrent('./mydb');
|
|
40
|
+
* console.log(`Concurrent: ${db.isConcurrent}`);
|
|
41
|
+
* ```
|
|
42
|
+
*
|
|
29
43
|
* @example Server Mode
|
|
30
44
|
* ```typescript
|
|
31
45
|
* import { SochDBClient } from '@sochdb/sochdb';
|
|
@@ -35,7 +49,7 @@
|
|
|
35
49
|
* await client.putKv('key', Buffer.from('value'));
|
|
36
50
|
* ```
|
|
37
51
|
*/
|
|
38
|
-
export declare const VERSION = "0.4.
|
|
52
|
+
export declare const VERSION = "0.4.4";
|
|
39
53
|
export { EmbeddedDatabase, EmbeddedDatabaseConfig } from './embedded';
|
|
40
54
|
export { EmbeddedTransaction } from './embedded';
|
|
41
55
|
export { HnswIndex, HnswConfig, HnswBindings } from './embedded';
|
|
@@ -49,8 +63,12 @@ export { SemanticCache, } from './semantic-cache';
|
|
|
49
63
|
export type { CacheEntry, CacheHit, CacheStats, } from './semantic-cache';
|
|
50
64
|
export { ContextQueryBuilder, ContextOutputFormat, TruncationStrategy, createContextBuilder, } from './context-builder';
|
|
51
65
|
export type { ContextResult, } from './context-builder';
|
|
52
|
-
export { ExtractionPipeline, Consolidator, HybridRetriever, AllowedSet,
|
|
53
|
-
export type { Entity, Relation, Assertion, RawAssertion, CanonicalFact, ExtractionResult, ExtractionSchema, ConsolidationConfig, RetrievalConfig, RetrievalResult, RetrievalResponse,
|
|
66
|
+
export { ExtractionPipeline, Consolidator, HybridRetriever, AllowedSet, } from './memory';
|
|
67
|
+
export type { Entity, Relation, Assertion, RawAssertion, CanonicalFact, ExtractionResult, ExtractionSchema, ConsolidationConfig, RetrievalConfig, RetrievalResult, RetrievalResponse, } from './memory';
|
|
68
|
+
export { McpServer, McpClient, McpError, MCP_ERROR_CODES, } from './mcp';
|
|
69
|
+
export type { McpTool, McpToolCall, McpToolResult, McpResource, McpResourceContent, McpPrompt, McpPromptArgument, McpPromptMessage, McpServerCapabilities, McpServerConfig, McpClientConfig, McpTransport, } from './mcp';
|
|
70
|
+
export { PolicyService, } from './policy';
|
|
71
|
+
export type { PolicyRule, PolicyCondition, PolicyEvaluation, NamespacePolicy, NamespaceGrant, NamespacePermission, PolicyAction, PolicyRequest, PolicySet, PolicyAuditEntry, } from './policy';
|
|
54
72
|
export { SochDBClient } from './grpc-client';
|
|
55
73
|
export type { SearchResult, Document, GraphNode, GraphEdge, } from './grpc-client';
|
|
56
74
|
export { IpcClient } from './ipc-client';
|