@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,26 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Policy Service Module
|
|
4
|
+
*
|
|
5
|
+
* Policy-based access control and namespace governance for SochDB.
|
|
6
|
+
*/
|
|
7
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
8
|
+
if (k2 === undefined) k2 = k;
|
|
9
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
10
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
11
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
12
|
+
}
|
|
13
|
+
Object.defineProperty(o, k2, desc);
|
|
14
|
+
}) : (function(o, m, k, k2) {
|
|
15
|
+
if (k2 === undefined) k2 = k;
|
|
16
|
+
o[k2] = m[k];
|
|
17
|
+
}));
|
|
18
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
19
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
20
|
+
};
|
|
21
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
22
|
+
exports.PolicyService = void 0;
|
|
23
|
+
__exportStar(require("./types"), exports);
|
|
24
|
+
var service_1 = require("./service");
|
|
25
|
+
Object.defineProperty(exports, "PolicyService", { enumerable: true, get: function () { return service_1.PolicyService; } });
|
|
26
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvcG9saWN5L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7OztHQUlHOzs7Ozs7Ozs7Ozs7Ozs7OztBQUVILDBDQUF3QjtBQUN4QixxQ0FBMEM7QUFBakMsd0dBQUEsYUFBYSxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBQb2xpY3kgU2VydmljZSBNb2R1bGVcbiAqIFxuICogUG9saWN5LWJhc2VkIGFjY2VzcyBjb250cm9sIGFuZCBuYW1lc3BhY2UgZ292ZXJuYW5jZSBmb3IgU29jaERCLlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuZXhwb3J0IHsgUG9saWN5U2VydmljZSB9IGZyb20gJy4vc2VydmljZSc7XG4iXX0=
|
|
@@ -0,0 +1,394 @@
|
|
|
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
|
+
constructor(db, options) {
|
|
43
|
+
this.cache = new Map();
|
|
44
|
+
this.auditEnabled = true;
|
|
45
|
+
this.db = db;
|
|
46
|
+
this.prefix = Buffer.from('_policy:');
|
|
47
|
+
this.auditEnabled = options?.enableAudit ?? true;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Create a namespace policy
|
|
51
|
+
*/
|
|
52
|
+
async createNamespacePolicy(policy) {
|
|
53
|
+
const key = this.policyKey(policy.namespace);
|
|
54
|
+
const data = {
|
|
55
|
+
...policy,
|
|
56
|
+
createdAt: Date.now(),
|
|
57
|
+
updatedAt: Date.now(),
|
|
58
|
+
};
|
|
59
|
+
await this.db.put(key, Buffer.from(JSON.stringify(data)));
|
|
60
|
+
this.cache.set(policy.namespace, policy);
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Get a namespace policy
|
|
64
|
+
*/
|
|
65
|
+
async getNamespacePolicy(namespace) {
|
|
66
|
+
// Check cache first
|
|
67
|
+
if (this.cache.has(namespace)) {
|
|
68
|
+
return this.cache.get(namespace);
|
|
69
|
+
}
|
|
70
|
+
const key = this.policyKey(namespace);
|
|
71
|
+
const value = await this.db.get(key);
|
|
72
|
+
if (!value) {
|
|
73
|
+
return null;
|
|
74
|
+
}
|
|
75
|
+
const policy = JSON.parse(value.toString());
|
|
76
|
+
this.cache.set(namespace, policy);
|
|
77
|
+
return policy;
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Update a namespace policy
|
|
81
|
+
*/
|
|
82
|
+
async updateNamespacePolicy(namespace, updates) {
|
|
83
|
+
const existing = await this.getNamespacePolicy(namespace);
|
|
84
|
+
if (!existing) {
|
|
85
|
+
throw new Error(`Policy not found for namespace: ${namespace}`);
|
|
86
|
+
}
|
|
87
|
+
const updated = {
|
|
88
|
+
...existing,
|
|
89
|
+
...updates,
|
|
90
|
+
namespace, // Ensure namespace doesn't change
|
|
91
|
+
updatedAt: Date.now(),
|
|
92
|
+
};
|
|
93
|
+
const key = this.policyKey(namespace);
|
|
94
|
+
await this.db.put(key, Buffer.from(JSON.stringify(updated)));
|
|
95
|
+
this.cache.set(namespace, updated);
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Delete a namespace policy
|
|
99
|
+
*/
|
|
100
|
+
async deleteNamespacePolicy(namespace) {
|
|
101
|
+
const key = this.policyKey(namespace);
|
|
102
|
+
await this.db.delete(key);
|
|
103
|
+
this.cache.delete(namespace);
|
|
104
|
+
return true;
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Add a rule to a namespace policy
|
|
108
|
+
*/
|
|
109
|
+
async addRule(namespace, rule) {
|
|
110
|
+
const policy = await this.getNamespacePolicy(namespace);
|
|
111
|
+
if (!policy) {
|
|
112
|
+
throw new Error(`Policy not found for namespace: ${namespace}`);
|
|
113
|
+
}
|
|
114
|
+
// Check for duplicate rule ID
|
|
115
|
+
if (policy.rules.some(r => r.id === rule.id)) {
|
|
116
|
+
throw new Error(`Rule with id '${rule.id}' already exists`);
|
|
117
|
+
}
|
|
118
|
+
policy.rules.push(rule);
|
|
119
|
+
await this.updateNamespacePolicy(namespace, { rules: policy.rules });
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Remove a rule from a namespace policy
|
|
123
|
+
*/
|
|
124
|
+
async removeRule(namespace, ruleId) {
|
|
125
|
+
const policy = await this.getNamespacePolicy(namespace);
|
|
126
|
+
if (!policy) {
|
|
127
|
+
return false;
|
|
128
|
+
}
|
|
129
|
+
const index = policy.rules.findIndex(r => r.id === ruleId);
|
|
130
|
+
if (index === -1) {
|
|
131
|
+
return false;
|
|
132
|
+
}
|
|
133
|
+
policy.rules.splice(index, 1);
|
|
134
|
+
await this.updateNamespacePolicy(namespace, { rules: policy.rules });
|
|
135
|
+
return true;
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Evaluate a policy request
|
|
139
|
+
*/
|
|
140
|
+
async evaluate(request) {
|
|
141
|
+
const startTime = Date.now();
|
|
142
|
+
// Extract namespace from resource
|
|
143
|
+
const namespace = this.extractNamespace(request.resource);
|
|
144
|
+
const policy = namespace ? await this.getNamespacePolicy(namespace) : null;
|
|
145
|
+
let result;
|
|
146
|
+
if (!policy) {
|
|
147
|
+
// No policy = allow by default
|
|
148
|
+
result = {
|
|
149
|
+
allowed: true,
|
|
150
|
+
reason: 'No policy defined',
|
|
151
|
+
evaluationTime: Date.now() - startTime,
|
|
152
|
+
};
|
|
153
|
+
}
|
|
154
|
+
else {
|
|
155
|
+
// Evaluate rules in priority order
|
|
156
|
+
const sortedRules = [...policy.rules].sort((a, b) => (a.priority || 0) - (b.priority || 0));
|
|
157
|
+
let matchedRule;
|
|
158
|
+
for (const rule of sortedRules) {
|
|
159
|
+
if (this.matchesRule(request, rule)) {
|
|
160
|
+
matchedRule = rule;
|
|
161
|
+
break;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
if (matchedRule) {
|
|
165
|
+
result = {
|
|
166
|
+
allowed: matchedRule.effect === 'allow',
|
|
167
|
+
matchedRule,
|
|
168
|
+
reason: `Matched rule: ${matchedRule.name}`,
|
|
169
|
+
evaluationTime: Date.now() - startTime,
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
else {
|
|
173
|
+
result = {
|
|
174
|
+
allowed: policy.defaultEffect === 'allow',
|
|
175
|
+
reason: `Default effect: ${policy.defaultEffect}`,
|
|
176
|
+
evaluationTime: Date.now() - startTime,
|
|
177
|
+
};
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
// Log audit entry
|
|
181
|
+
if (this.auditEnabled) {
|
|
182
|
+
await this.logAudit(request, result);
|
|
183
|
+
}
|
|
184
|
+
return result;
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* Grant namespace access to a principal
|
|
188
|
+
*/
|
|
189
|
+
async grantAccess(grant) {
|
|
190
|
+
const key = this.grantKey(grant.namespace, grant.principal);
|
|
191
|
+
const data = {
|
|
192
|
+
...grant,
|
|
193
|
+
grantedAt: Date.now(),
|
|
194
|
+
};
|
|
195
|
+
await this.db.put(key, Buffer.from(JSON.stringify(data)));
|
|
196
|
+
}
|
|
197
|
+
/**
|
|
198
|
+
* Revoke namespace access from a principal
|
|
199
|
+
*/
|
|
200
|
+
async revokeAccess(namespace, principal) {
|
|
201
|
+
const key = this.grantKey(namespace, principal);
|
|
202
|
+
await this.db.delete(key);
|
|
203
|
+
return true;
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Check if principal has permission
|
|
207
|
+
*/
|
|
208
|
+
async hasPermission(namespace, principal, permission) {
|
|
209
|
+
const key = this.grantKey(namespace, principal);
|
|
210
|
+
const value = await this.db.get(key);
|
|
211
|
+
if (!value) {
|
|
212
|
+
return false;
|
|
213
|
+
}
|
|
214
|
+
const grant = JSON.parse(value.toString());
|
|
215
|
+
// Check expiration
|
|
216
|
+
if (grant.expiresAt && Date.now() > grant.expiresAt) {
|
|
217
|
+
return false;
|
|
218
|
+
}
|
|
219
|
+
// Admin has all permissions
|
|
220
|
+
if (grant.permissions.includes('admin')) {
|
|
221
|
+
return true;
|
|
222
|
+
}
|
|
223
|
+
return grant.permissions.includes(permission);
|
|
224
|
+
}
|
|
225
|
+
/**
|
|
226
|
+
* List all grants for a namespace
|
|
227
|
+
*/
|
|
228
|
+
async listGrants(namespace) {
|
|
229
|
+
const grants = [];
|
|
230
|
+
const prefix = Buffer.from(`_grant:${namespace}:`);
|
|
231
|
+
try {
|
|
232
|
+
for await (const [_, valueBuffer] of this.db.scanPrefix(prefix)) {
|
|
233
|
+
const grant = JSON.parse(valueBuffer.toString());
|
|
234
|
+
grants.push(grant);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
catch (error) {
|
|
238
|
+
// Ignore scan errors
|
|
239
|
+
}
|
|
240
|
+
return grants;
|
|
241
|
+
}
|
|
242
|
+
/**
|
|
243
|
+
* Get audit log entries
|
|
244
|
+
*/
|
|
245
|
+
async getAuditLog(options) {
|
|
246
|
+
const entries = [];
|
|
247
|
+
const prefix = Buffer.from('_audit:');
|
|
248
|
+
const limit = options?.limit || 100;
|
|
249
|
+
try {
|
|
250
|
+
for await (const [_, valueBuffer] of this.db.scanPrefix(prefix)) {
|
|
251
|
+
const entry = JSON.parse(valueBuffer.toString());
|
|
252
|
+
// Apply filters
|
|
253
|
+
if (options?.namespace && !entry.resource.includes(options.namespace)) {
|
|
254
|
+
continue;
|
|
255
|
+
}
|
|
256
|
+
if (options?.principal && entry.principal !== options.principal) {
|
|
257
|
+
continue;
|
|
258
|
+
}
|
|
259
|
+
if (options?.action && entry.action !== options.action) {
|
|
260
|
+
continue;
|
|
261
|
+
}
|
|
262
|
+
if (options?.since && entry.timestamp < options.since) {
|
|
263
|
+
continue;
|
|
264
|
+
}
|
|
265
|
+
entries.push(entry);
|
|
266
|
+
if (entries.length >= limit) {
|
|
267
|
+
break;
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
catch (error) {
|
|
272
|
+
// Ignore scan errors
|
|
273
|
+
}
|
|
274
|
+
// Sort by timestamp descending
|
|
275
|
+
return entries.sort((a, b) => b.timestamp - a.timestamp);
|
|
276
|
+
}
|
|
277
|
+
/**
|
|
278
|
+
* Clear policy cache
|
|
279
|
+
*/
|
|
280
|
+
clearCache() {
|
|
281
|
+
this.cache.clear();
|
|
282
|
+
}
|
|
283
|
+
// Private methods
|
|
284
|
+
policyKey(namespace) {
|
|
285
|
+
return Buffer.concat([this.prefix, Buffer.from(`namespace:${namespace}`)]);
|
|
286
|
+
}
|
|
287
|
+
grantKey(namespace, principal) {
|
|
288
|
+
return Buffer.from(`_grant:${namespace}:${principal}`);
|
|
289
|
+
}
|
|
290
|
+
extractNamespace(resource) {
|
|
291
|
+
// Extract namespace from resource like "namespace:tenant_123:collection:docs"
|
|
292
|
+
const parts = resource.split(':');
|
|
293
|
+
if (parts.length >= 2 && parts[0] === 'namespace') {
|
|
294
|
+
return parts[1];
|
|
295
|
+
}
|
|
296
|
+
// Try to extract from collection format "collection:tenant_123:docs"
|
|
297
|
+
if (parts.length >= 2 && parts[0] === 'collection') {
|
|
298
|
+
return parts[1];
|
|
299
|
+
}
|
|
300
|
+
return null;
|
|
301
|
+
}
|
|
302
|
+
matchesRule(request, rule) {
|
|
303
|
+
// Check principals
|
|
304
|
+
if (!this.matchesPatterns(request.principal, rule.principals)) {
|
|
305
|
+
return false;
|
|
306
|
+
}
|
|
307
|
+
// Check resources
|
|
308
|
+
if (!this.matchesPatterns(request.resource, rule.resources)) {
|
|
309
|
+
return false;
|
|
310
|
+
}
|
|
311
|
+
// Check actions
|
|
312
|
+
if (!this.matchesPatterns(request.action, rule.actions)) {
|
|
313
|
+
return false;
|
|
314
|
+
}
|
|
315
|
+
// Check conditions
|
|
316
|
+
if (rule.conditions && rule.conditions.length > 0) {
|
|
317
|
+
for (const condition of rule.conditions) {
|
|
318
|
+
if (!this.evaluateCondition(condition, request.context)) {
|
|
319
|
+
return false;
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
}
|
|
323
|
+
return true;
|
|
324
|
+
}
|
|
325
|
+
matchesPatterns(value, patterns) {
|
|
326
|
+
for (const pattern of patterns) {
|
|
327
|
+
if (this.matchesPattern(value, pattern)) {
|
|
328
|
+
return true;
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
return false;
|
|
332
|
+
}
|
|
333
|
+
matchesPattern(value, pattern) {
|
|
334
|
+
if (pattern === '*') {
|
|
335
|
+
return true;
|
|
336
|
+
}
|
|
337
|
+
if (pattern.endsWith('*')) {
|
|
338
|
+
return value.startsWith(pattern.slice(0, -1));
|
|
339
|
+
}
|
|
340
|
+
if (pattern.startsWith('*')) {
|
|
341
|
+
return value.endsWith(pattern.slice(1));
|
|
342
|
+
}
|
|
343
|
+
return value === pattern;
|
|
344
|
+
}
|
|
345
|
+
evaluateCondition(condition, context) {
|
|
346
|
+
if (!context) {
|
|
347
|
+
return false;
|
|
348
|
+
}
|
|
349
|
+
const value = context[condition.key];
|
|
350
|
+
if (value === undefined) {
|
|
351
|
+
return false;
|
|
352
|
+
}
|
|
353
|
+
switch (condition.operator) {
|
|
354
|
+
case 'equals':
|
|
355
|
+
return value === condition.value;
|
|
356
|
+
case 'not_equals':
|
|
357
|
+
return value !== condition.value;
|
|
358
|
+
case 'contains':
|
|
359
|
+
return String(value).includes(String(condition.value));
|
|
360
|
+
case 'starts_with':
|
|
361
|
+
return String(value).startsWith(String(condition.value));
|
|
362
|
+
case 'ends_with':
|
|
363
|
+
return String(value).endsWith(String(condition.value));
|
|
364
|
+
case 'in':
|
|
365
|
+
return Array.isArray(condition.value) && condition.value.includes(value);
|
|
366
|
+
case 'not_in':
|
|
367
|
+
return Array.isArray(condition.value) && !condition.value.includes(value);
|
|
368
|
+
case 'between':
|
|
369
|
+
if (Array.isArray(condition.value) && condition.value.length === 2) {
|
|
370
|
+
return value >= condition.value[0] && value <= condition.value[1];
|
|
371
|
+
}
|
|
372
|
+
return false;
|
|
373
|
+
default:
|
|
374
|
+
return false;
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
async logAudit(request, result) {
|
|
378
|
+
const entry = {
|
|
379
|
+
id: `${Date.now()}-${Math.random().toString(36).substr(2, 9)}`,
|
|
380
|
+
timestamp: Date.now(),
|
|
381
|
+
principal: request.principal,
|
|
382
|
+
action: request.action,
|
|
383
|
+
resource: request.resource,
|
|
384
|
+
decision: result.allowed ? 'allow' : 'deny',
|
|
385
|
+
matchedRule: result.matchedRule?.id,
|
|
386
|
+
reason: result.reason,
|
|
387
|
+
context: request.context,
|
|
388
|
+
};
|
|
389
|
+
const key = Buffer.from(`_audit:${entry.timestamp}:${entry.id}`);
|
|
390
|
+
await this.db.put(key, Buffer.from(JSON.stringify(entry)));
|
|
391
|
+
}
|
|
392
|
+
}
|
|
393
|
+
exports.PolicyService = PolicyService;
|
|
394
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service.js","sourceRoot":"","sources":["../../../src/policy/service.ts"],"names":[],"mappings":";AAAA;;;;GAIG;;;AAeH;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BG;AACH,MAAa,aAAa;IAMxB,YAAY,EAAoB,EAAE,OAAmC;QAH7D,UAAK,GAAiC,IAAI,GAAG,EAAE,CAAC;QAChD,iBAAY,GAAG,IAAI,CAAC;QAG1B,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;QACb,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACtC,IAAI,CAAC,YAAY,GAAG,OAAO,EAAE,WAAW,IAAI,IAAI,CAAC;IACnD,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,qBAAqB,CAAC,MAAuB;QACjD,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAC7C,MAAM,IAAI,GAAG;YACX,GAAG,MAAM;YACT,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;YACrB,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;SACtB,CAAC;QAEF,MAAM,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAC1D,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,kBAAkB,CAAC,SAAiB;QACxC,oBAAoB;QACpB,IAAI,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAE,CAAC;QACpC,CAAC;QAED,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;QACtC,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAErC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAoB,CAAC;QAC/D,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;QAClC,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,qBAAqB,CAAC,SAAiB,EAAE,OAAiC;QAC9E,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAC;QAC1D,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,MAAM,IAAI,KAAK,CAAC,mCAAmC,SAAS,EAAE,CAAC,CAAC;QAClE,CAAC;QAED,MAAM,OAAO,GAAG;YACd,GAAG,QAAQ;YACX,GAAG,OAAO;YACV,SAAS,EAAE,kCAAkC;YAC7C,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;SACtB,CAAC;QAEF,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;QACtC,MAAM,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QAC7D,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;IACrC,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,qBAAqB,CAAC,SAAiB;QAC3C,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;QACtC,MAAM,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QAC1B,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAC7B,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,SAAiB,EAAE,IAAgB;QAC/C,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAC;QACxD,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,MAAM,IAAI,KAAK,CAAC,mCAAmC,SAAS,EAAE,CAAC,CAAC;QAClE,CAAC;QAED,8BAA8B;QAC9B,IAAI,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC;YAC7C,MAAM,IAAI,KAAK,CAAC,iBAAiB,IAAI,CAAC,EAAE,kBAAkB,CAAC,CAAC;QAC9D,CAAC;QAED,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACxB,MAAM,IAAI,CAAC,qBAAqB,CAAC,SAAS,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;IACvE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,UAAU,CAAC,SAAiB,EAAE,MAAc;QAChD,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAC;QACxD,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,MAAM,CAAC,CAAC;QAC3D,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE,CAAC;YACjB,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QAC9B,MAAM,IAAI,CAAC,qBAAqB,CAAC,SAAS,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;QACrE,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,QAAQ,CAAC,OAAsB;QACnC,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAE7B,kCAAkC;QAClC,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAC1D,MAAM,MAAM,GAAG,SAAS,CAAC,CAAC,CAAC,MAAM,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAE3E,IAAI,MAAwB,CAAC;QAE7B,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,+BAA+B;YAC/B,MAAM,GAAG;gBACP,OAAO,EAAE,IAAI;gBACb,MAAM,EAAE,mBAAmB;gBAC3B,cAAc,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS;aACvC,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,mCAAmC;YACnC,MAAM,WAAW,GAAG,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAClD,CAAC,CAAC,CAAC,QAAQ,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,IAAI,CAAC,CAAC,CACtC,CAAC;YAEF,IAAI,WAAmC,CAAC;YAExC,KAAK,MAAM,IAAI,IAAI,WAAW,EAAE,CAAC;gBAC/B,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,IAAI,CAAC,EAAE,CAAC;oBACpC,WAAW,GAAG,IAAI,CAAC;oBACnB,MAAM;gBACR,CAAC;YACH,CAAC;YAED,IAAI,WAAW,EAAE,CAAC;gBAChB,MAAM,GAAG;oBACP,OAAO,EAAE,WAAW,CAAC,MAAM,KAAK,OAAO;oBACvC,WAAW;oBACX,MAAM,EAAE,iBAAiB,WAAW,CAAC,IAAI,EAAE;oBAC3C,cAAc,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS;iBACvC,CAAC;YACJ,CAAC;iBAAM,CAAC;gBACN,MAAM,GAAG;oBACP,OAAO,EAAE,MAAM,CAAC,aAAa,KAAK,OAAO;oBACzC,MAAM,EAAE,mBAAmB,MAAM,CAAC,aAAa,EAAE;oBACjD,cAAc,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS;iBACvC,CAAC;YACJ,CAAC;QACH,CAAC;QAED,kBAAkB;QAClB,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,MAAM,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QACvC,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW,CAAC,KAAqB;QACrC,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;QAC5D,MAAM,IAAI,GAAG;YACX,GAAG,KAAK;YACR,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;SACtB,CAAC;QAEF,MAAM,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,YAAY,CAAC,SAAiB,EAAE,SAAiB;QACrD,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;QAChD,MAAM,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QAC1B,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,aAAa,CACjB,SAAiB,EACjB,SAAiB,EACjB,UAA+B;QAE/B,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;QAChD,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAErC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAmB,CAAC;QAE7D,mBAAmB;QACnB,IAAI,KAAK,CAAC,SAAS,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC,SAAS,EAAE,CAAC;YACpD,OAAO,KAAK,CAAC;QACf,CAAC;QAED,4BAA4B;QAC5B,IAAI,KAAK,CAAC,WAAW,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACxC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,KAAK,CAAC,WAAW,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;IAChD,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,UAAU,CAAC,SAAiB;QAChC,MAAM,MAAM,GAAqB,EAAE,CAAC;QACpC,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,SAAS,GAAG,CAAC,CAAC;QAEnD,IAAI,CAAC;YACH,IAAI,KAAK,EAAE,MAAM,CAAC,CAAC,EAAE,WAAW,CAAC,IAAI,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;gBAChE,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAmB,CAAC;gBACnE,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACrB,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,qBAAqB;QACvB,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW,CAAC,OAMjB;QACC,MAAM,OAAO,GAAuB,EAAE,CAAC;QACvC,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACtC,MAAM,KAAK,GAAG,OAAO,EAAE,KAAK,IAAI,GAAG,CAAC;QAEpC,IAAI,CAAC;YACH,IAAI,KAAK,EAAE,MAAM,CAAC,CAAC,EAAE,WAAW,CAAC,IAAI,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;gBAChE,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAqB,CAAC;gBAErE,gBAAgB;gBAChB,IAAI,OAAO,EAAE,SAAS,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;oBACtE,SAAS;gBACX,CAAC;gBACD,IAAI,OAAO,EAAE,SAAS,IAAI,KAAK,CAAC,SAAS,KAAK,OAAO,CAAC,SAAS,EAAE,CAAC;oBAChE,SAAS;gBACX,CAAC;gBACD,IAAI,OAAO,EAAE,MAAM,IAAI,KAAK,CAAC,MAAM,KAAK,OAAO,CAAC,MAAM,EAAE,CAAC;oBACvD,SAAS;gBACX,CAAC;gBACD,IAAI,OAAO,EAAE,KAAK,IAAI,KAAK,CAAC,SAAS,GAAG,OAAO,CAAC,KAAK,EAAE,CAAC;oBACtD,SAAS;gBACX,CAAC;gBAED,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAEpB,IAAI,OAAO,CAAC,MAAM,IAAI,KAAK,EAAE,CAAC;oBAC5B,MAAM;gBACR,CAAC;YACH,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,qBAAqB;QACvB,CAAC;QAED,+BAA+B;QAC/B,OAAO,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC,SAAS,CAAC,CAAC;IAC3D,CAAC;IAED;;OAEG;IACH,UAAU;QACR,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;IACrB,CAAC;IAED,kBAAkB;IAEV,SAAS,CAAC,SAAiB;QACjC,OAAO,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,aAAa,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;IAC7E,CAAC;IAEO,QAAQ,CAAC,SAAiB,EAAE,SAAiB;QACnD,OAAO,MAAM,CAAC,IAAI,CAAC,UAAU,SAAS,IAAI,SAAS,EAAE,CAAC,CAAC;IACzD,CAAC;IAEO,gBAAgB,CAAC,QAAgB;QACvC,8EAA8E;QAC9E,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAClC,IAAI,KAAK,CAAC,MAAM,IAAI,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC,KAAK,WAAW,EAAE,CAAC;YAClD,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QACD,qEAAqE;QACrE,IAAI,KAAK,CAAC,MAAM,IAAI,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC,KAAK,YAAY,EAAE,CAAC;YACnD,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,WAAW,CAAC,OAAsB,EAAE,IAAgB;QAC1D,mBAAmB;QACnB,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,SAAS,EAAE,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YAC9D,OAAO,KAAK,CAAC;QACf,CAAC;QAED,kBAAkB;QAClB,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC;YAC5D,OAAO,KAAK,CAAC;QACf,CAAC;QAED,gBAAgB;QAChB,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YACxD,OAAO,KAAK,CAAC;QACf,CAAC;QAED,mBAAmB;QACnB,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAClD,KAAK,MAAM,SAAS,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBACxC,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,SAAS,EAAE,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;oBACxD,OAAO,KAAK,CAAC;gBACf,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,eAAe,CAAC,KAAa,EAAE,QAAkB;QACvD,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;YAC/B,IAAI,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,OAAO,CAAC,EAAE,CAAC;gBACxC,OAAO,IAAI,CAAC;YACd,CAAC;QACH,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,cAAc,CAAC,KAAa,EAAE,OAAe;QACnD,IAAI,OAAO,KAAK,GAAG,EAAE,CAAC;YACpB,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IAAI,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;YAC1B,OAAO,KAAK,CAAC,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QAChD,CAAC;QAED,IAAI,OAAO,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YAC5B,OAAO,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QAC1C,CAAC;QAED,OAAO,KAAK,KAAK,OAAO,CAAC;IAC3B,CAAC;IAEO,iBAAiB,CAAC,SAA0B,EAAE,OAA6B;QACjF,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,KAAK,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YACxB,OAAO,KAAK,CAAC;QACf,CAAC;QAED,QAAQ,SAAS,CAAC,QAAQ,EAAE,CAAC;YAC3B,KAAK,QAAQ;gBACX,OAAO,KAAK,KAAK,SAAS,CAAC,KAAK,CAAC;YACnC,KAAK,YAAY;gBACf,OAAO,KAAK,KAAK,SAAS,CAAC,KAAK,CAAC;YACnC,KAAK,UAAU;gBACb,OAAO,MAAM,CAAC,KAAK,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;YACzD,KAAK,aAAa;gBAChB,OAAO,MAAM,CAAC,KAAK,CAAC,CAAC,UAAU,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;YAC3D,KAAK,WAAW;gBACd,OAAO,MAAM,CAAC,KAAK,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;YACzD,KAAK,IAAI;gBACP,OAAO,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,SAAS,CAAC,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YAC3E,KAAK,QAAQ;gBACX,OAAO,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YAC5E,KAAK,SAAS;gBACZ,IAAI,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,SAAS,CAAC,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBACnE,OAAO,KAAK,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,KAAK,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBACpE,CAAC;gBACD,OAAO,KAAK,CAAC;YACf;gBACE,OAAO,KAAK,CAAC;QACjB,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,QAAQ,CAAC,OAAsB,EAAE,MAAwB;QACrE,MAAM,KAAK,GAAqB;YAC9B,EAAE,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE;YAC9D,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;YACrB,SAAS,EAAE,OAAO,CAAC,SAAS;YAC5B,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,QAAQ,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM;YAC3C,WAAW,EAAE,MAAM,CAAC,WAAW,EAAE,EAAE;YACnC,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,OAAO,EAAE,OAAO,CAAC,OAAO;SACzB,CAAC;QAEF,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,KAAK,CAAC,SAAS,IAAI,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC;QACjE,MAAM,IAAI,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC7D,CAAC;CACF;AAzaD,sCAyaC","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
|