trellis 2.1.7 → 2.1.9
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 +100 -10
- package/dist/cli/index.d.ts +14 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +199 -9
- package/dist/client/config.d.ts +56 -0
- package/dist/client/config.d.ts.map +1 -0
- package/dist/client/index.d.ts +15 -0
- package/dist/client/index.d.ts.map +1 -0
- package/dist/client/index.js +32 -0
- package/dist/client/sdk.d.ts +152 -0
- package/dist/client/sdk.d.ts.map +1 -0
- package/dist/config-8hczw0rs.js +20 -0
- package/dist/core/agents/harness.d.ts +49 -0
- package/dist/core/agents/harness.d.ts.map +1 -0
- package/dist/core/agents/index.d.ts +8 -0
- package/dist/core/agents/index.d.ts.map +1 -0
- package/dist/core/agents/types.d.ts +63 -0
- package/dist/core/agents/types.d.ts.map +1 -0
- package/dist/core/index.d.ts +25 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +15 -9
- package/dist/core/kernel/middleware.d.ts +28 -0
- package/dist/core/kernel/middleware.d.ts.map +1 -0
- package/dist/core/kernel/trellis-kernel.d.ts +162 -0
- package/dist/core/kernel/trellis-kernel.d.ts.map +1 -0
- package/dist/core/ontology/builtins.d.ts +16 -0
- package/dist/core/ontology/builtins.d.ts.map +1 -0
- package/dist/core/ontology/index.d.ts +10 -0
- package/dist/core/ontology/index.d.ts.map +1 -0
- package/dist/core/ontology/registry.d.ts +70 -0
- package/dist/core/ontology/registry.d.ts.map +1 -0
- package/dist/core/ontology/types.d.ts +97 -0
- package/dist/core/ontology/types.d.ts.map +1 -0
- package/dist/core/ontology/validator.d.ts +34 -0
- package/dist/core/ontology/validator.d.ts.map +1 -0
- package/dist/core/persist/backend.d.ts +60 -0
- package/dist/core/persist/backend.d.ts.map +1 -0
- package/dist/core/persist/sqlite-backend.d.ts +41 -0
- package/dist/core/persist/sqlite-backend.d.ts.map +1 -0
- package/dist/core/plugins/index.d.ts +8 -0
- package/dist/core/plugins/index.d.ts.map +1 -0
- package/dist/core/plugins/registry.d.ts +69 -0
- package/dist/core/plugins/registry.d.ts.map +1 -0
- package/dist/core/plugins/types.d.ts +87 -0
- package/dist/core/plugins/types.d.ts.map +1 -0
- package/dist/core/query/datalog.d.ts +52 -0
- package/dist/core/query/datalog.d.ts.map +1 -0
- package/dist/core/query/engine.d.ts +42 -0
- package/dist/core/query/engine.d.ts.map +1 -0
- package/dist/core/query/index.d.ts +12 -0
- package/dist/core/query/index.d.ts.map +1 -0
- package/dist/core/query/parser.d.ts +37 -0
- package/dist/core/query/parser.d.ts.map +1 -0
- package/dist/core/query/types.d.ts +135 -0
- package/dist/core/query/types.d.ts.map +1 -0
- package/dist/core/store/eav-store.d.ts +107 -0
- package/dist/core/store/eav-store.d.ts.map +1 -0
- package/dist/db/index.d.ts +18 -0
- package/dist/db/index.d.ts.map +1 -0
- package/dist/db/index.js +83 -0
- package/dist/db/inspector.js +28 -0
- package/dist/db/trellis.css +1 -0
- package/dist/decisions/auto-capture.d.ts +31 -0
- package/dist/decisions/auto-capture.d.ts.map +1 -0
- package/dist/decisions/hooks.d.ts +48 -0
- package/dist/decisions/hooks.d.ts.map +1 -0
- package/dist/decisions/index.d.ts +36 -0
- package/dist/decisions/index.d.ts.map +1 -0
- package/dist/decisions/types.d.ts +73 -0
- package/dist/decisions/types.d.ts.map +1 -0
- package/dist/deploy-99j5dc9c.js +9 -0
- package/dist/embeddings/auto-embed.d.ts +52 -0
- package/dist/embeddings/auto-embed.d.ts.map +1 -0
- package/dist/embeddings/chunker.d.ts +73 -0
- package/dist/embeddings/chunker.d.ts.map +1 -0
- package/dist/embeddings/index.d.ts +18 -0
- package/dist/embeddings/index.d.ts.map +1 -0
- package/dist/embeddings/model.d.ts +30 -0
- package/dist/embeddings/model.d.ts.map +1 -0
- package/dist/embeddings/search.d.ts +87 -0
- package/dist/embeddings/search.d.ts.map +1 -0
- package/dist/embeddings/store.d.ts +66 -0
- package/dist/embeddings/store.d.ts.map +1 -0
- package/dist/embeddings/types.d.ts +54 -0
- package/dist/embeddings/types.d.ts.map +1 -0
- package/dist/engine-y0724kjq.js +8 -0
- package/dist/engine.d.ts +216 -0
- package/dist/engine.d.ts.map +1 -0
- package/dist/garden/cluster.d.ts +57 -0
- package/dist/garden/cluster.d.ts.map +1 -0
- package/dist/garden/garden.d.ts +104 -0
- package/dist/garden/garden.d.ts.map +1 -0
- package/dist/garden/index.d.ts +15 -0
- package/dist/garden/index.d.ts.map +1 -0
- package/dist/git/git-exporter.d.ts +37 -0
- package/dist/git/git-exporter.d.ts.map +1 -0
- package/dist/git/git-importer.d.ts +36 -0
- package/dist/git/git-importer.d.ts.map +1 -0
- package/dist/git/git-reader.d.ts +63 -0
- package/dist/git/git-reader.d.ts.map +1 -0
- package/dist/git/index.d.ts +10 -0
- package/dist/git/index.d.ts.map +1 -0
- package/dist/identity/governance.d.ts +54 -0
- package/dist/identity/governance.d.ts.map +1 -0
- package/dist/identity/identity.d.ts +63 -0
- package/dist/identity/identity.d.ts.map +1 -0
- package/dist/identity/index.d.ts +10 -0
- package/dist/identity/index.d.ts.map +1 -0
- package/dist/identity/signing-middleware.d.ts +38 -0
- package/dist/identity/signing-middleware.d.ts.map +1 -0
- package/dist/import-fyg5sgq4.js +11 -0
- package/dist/{index-5m0g9r0y.js → index-04sq3h27.js} +1 -496
- package/dist/index-2r4jxwnb.js +32 -0
- package/dist/index-4beszbgg.js +292 -0
- package/dist/index-5bhe57y9.js +326 -0
- package/dist/index-5p6zgspx.js +77 -0
- package/dist/index-7t92ej34.js +544 -0
- package/dist/index-8fjwnztt.js +155 -0
- package/dist/{index-22jx9qsz.js → index-czecrvvn.js} +4 -4
- package/dist/index-hgd30epa.js +149 -0
- package/dist/index-k5b0xskw.js +1 -0
- package/dist/index-kbnht9p8.js +1 -0
- package/dist/index-n9f2qyh5.js +495 -0
- package/dist/index-xr7rx360.js +814 -0
- package/dist/index-xzym9w0m.js +43 -0
- package/dist/index-yp88he8n.js +316 -0
- package/dist/index.d.ts +23 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +9 -7
- package/dist/links/index.d.ts +16 -0
- package/dist/links/index.d.ts.map +1 -0
- package/dist/links/lifecycle.d.ts +112 -0
- package/dist/links/lifecycle.d.ts.map +1 -0
- package/dist/links/parser.d.ts +56 -0
- package/dist/links/parser.d.ts.map +1 -0
- package/dist/links/ref-index.d.ts +55 -0
- package/dist/links/ref-index.d.ts.map +1 -0
- package/dist/links/resolver.d.ts +90 -0
- package/dist/links/resolver.d.ts.map +1 -0
- package/dist/links/types.d.ts +70 -0
- package/dist/links/types.d.ts.map +1 -0
- package/dist/mcp/docs.d.ts +18 -0
- package/dist/mcp/docs.d.ts.map +1 -0
- package/dist/mcp/index.d.ts +9 -0
- package/dist/mcp/index.d.ts.map +1 -0
- package/dist/mcp/server.d.ts +26 -0
- package/dist/mcp/server.d.ts.map +1 -0
- package/dist/react/index.js +189 -0
- package/dist/sdk-sj8rp0m7.js +15 -0
- package/dist/semantic/csharp-parser.d.ts +12 -0
- package/dist/semantic/csharp-parser.d.ts.map +1 -0
- package/dist/semantic/go-parser.d.ts +12 -0
- package/dist/semantic/go-parser.d.ts.map +1 -0
- package/dist/semantic/index.d.ts +22 -0
- package/dist/semantic/index.d.ts.map +1 -0
- package/dist/semantic/java-parser.d.ts +12 -0
- package/dist/semantic/java-parser.d.ts.map +1 -0
- package/dist/semantic/python-parser.d.ts +12 -0
- package/dist/semantic/python-parser.d.ts.map +1 -0
- package/dist/semantic/ruby-parser.d.ts +12 -0
- package/dist/semantic/ruby-parser.d.ts.map +1 -0
- package/dist/semantic/rust-parser.d.ts +12 -0
- package/dist/semantic/rust-parser.d.ts.map +1 -0
- package/dist/semantic/semantic-merge.d.ts +20 -0
- package/dist/semantic/semantic-merge.d.ts.map +1 -0
- package/dist/semantic/ts-parser.d.ts +13 -0
- package/dist/semantic/ts-parser.d.ts.map +1 -0
- package/dist/semantic/types.d.ts +130 -0
- package/dist/semantic/types.d.ts.map +1 -0
- package/dist/server/auth.d.ts +72 -0
- package/dist/server/auth.d.ts.map +1 -0
- package/dist/server/deploy.d.ts +44 -0
- package/dist/server/deploy.d.ts.map +1 -0
- package/dist/server/import.d.ts +40 -0
- package/dist/server/import.d.ts.map +1 -0
- package/dist/server/index.d.ts +24 -0
- package/dist/server/index.d.ts.map +1 -0
- package/dist/server/index.js +60 -0
- package/dist/server/permissions.d.ts +84 -0
- package/dist/server/permissions.d.ts.map +1 -0
- package/dist/server/realtime.d.ts +78 -0
- package/dist/server/realtime.d.ts.map +1 -0
- package/dist/server/server.d.ts +43 -0
- package/dist/server/server.d.ts.map +1 -0
- package/dist/server/tenancy.d.ts +53 -0
- package/dist/server/tenancy.d.ts.map +1 -0
- package/dist/server-3vkpnpbz.js +11 -0
- package/dist/sync/http-transport.d.ts +47 -0
- package/dist/sync/http-transport.d.ts.map +1 -0
- package/dist/sync/index.d.ts +22 -0
- package/dist/sync/index.d.ts.map +1 -0
- package/dist/sync/memory-transport.d.ts +27 -0
- package/dist/sync/memory-transport.d.ts.map +1 -0
- package/dist/sync/multi-repo.d.ts +82 -0
- package/dist/sync/multi-repo.d.ts.map +1 -0
- package/dist/sync/reconciler.d.ts +48 -0
- package/dist/sync/reconciler.d.ts.map +1 -0
- package/dist/sync/sync-engine.d.ts +65 -0
- package/dist/sync/sync-engine.d.ts.map +1 -0
- package/dist/sync/types.d.ts +71 -0
- package/dist/sync/types.d.ts.map +1 -0
- package/dist/sync/ws-transport.d.ts +41 -0
- package/dist/sync/ws-transport.d.ts.map +1 -0
- package/dist/tenancy-tjr7kk2v.js +13 -0
- package/dist/ui/server.d.ts +38 -0
- package/dist/ui/server.d.ts.map +1 -0
- package/dist/vcs/blob-store.d.ts +49 -0
- package/dist/vcs/blob-store.d.ts.map +1 -0
- package/dist/vcs/branch.d.ts +35 -0
- package/dist/vcs/branch.d.ts.map +1 -0
- package/dist/vcs/checkpoint.d.ts +24 -0
- package/dist/vcs/checkpoint.d.ts.map +1 -0
- package/dist/vcs/decompose.d.ts +19 -0
- package/dist/vcs/decompose.d.ts.map +1 -0
- package/dist/vcs/diff.d.ts +65 -0
- package/dist/vcs/diff.d.ts.map +1 -0
- package/dist/vcs/engine-context.d.ts +21 -0
- package/dist/vcs/engine-context.d.ts.map +1 -0
- package/dist/vcs/index.d.ts +23 -0
- package/dist/vcs/index.d.ts.map +1 -0
- package/dist/vcs/issue.d.ts +159 -0
- package/dist/vcs/issue.d.ts.map +1 -0
- package/dist/vcs/merge.d.ts +55 -0
- package/dist/vcs/merge.d.ts.map +1 -0
- package/dist/vcs/milestone.d.ts +30 -0
- package/dist/vcs/milestone.d.ts.map +1 -0
- package/dist/vcs/ops.d.ts +27 -0
- package/dist/vcs/ops.d.ts.map +1 -0
- package/dist/vcs/types.d.ts +94 -0
- package/dist/vcs/types.d.ts.map +1 -0
- package/dist/vcs/vcs-middleware.d.ts +14 -0
- package/dist/vcs/vcs-middleware.d.ts.map +1 -0
- package/dist/watcher/fs-watcher.d.ts +51 -0
- package/dist/watcher/fs-watcher.d.ts.map +1 -0
- package/dist/watcher/index.d.ts +9 -0
- package/dist/watcher/index.d.ts.map +1 -0
- package/dist/watcher/ingestion.d.ts +28 -0
- package/dist/watcher/ingestion.d.ts.map +1 -0
- package/package.json +44 -4
- package/dist/index-hybgxe40.js +0 -1174
|
@@ -0,0 +1,292 @@
|
|
|
1
|
+
// @bun
|
|
2
|
+
import {
|
|
3
|
+
readConfig
|
|
4
|
+
} from "./index-xzym9w0m.js";
|
|
5
|
+
import {
|
|
6
|
+
TenantPool
|
|
7
|
+
} from "./index-5p6zgspx.js";
|
|
8
|
+
import {
|
|
9
|
+
__require
|
|
10
|
+
} from "./index-a76rekgs.js";
|
|
11
|
+
|
|
12
|
+
// src/client/sdk.ts
|
|
13
|
+
function isRemote(opts) {
|
|
14
|
+
return "url" in opts;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
class TrellisDb {
|
|
18
|
+
opts;
|
|
19
|
+
_pool = null;
|
|
20
|
+
_ws = null;
|
|
21
|
+
_subCallbacks = new Map;
|
|
22
|
+
constructor(opts) {
|
|
23
|
+
this.opts = opts;
|
|
24
|
+
}
|
|
25
|
+
static fromConfig(dir = ".") {
|
|
26
|
+
const config = readConfig(dir);
|
|
27
|
+
if (!config)
|
|
28
|
+
throw new Error("No .trellis-db.json found. Run `trellis db init` first.");
|
|
29
|
+
if (config.mode === "remote" && config.url) {
|
|
30
|
+
return new TrellisDb({ url: config.url, apiKey: config.apiKey });
|
|
31
|
+
}
|
|
32
|
+
if (config.dbPath) {
|
|
33
|
+
return new TrellisDb({ path: config.dbPath });
|
|
34
|
+
}
|
|
35
|
+
throw new Error("Invalid .trellis-db.json: missing url or dbPath.");
|
|
36
|
+
}
|
|
37
|
+
async create(type, attributes = {}, links) {
|
|
38
|
+
if (isRemote(this.opts)) {
|
|
39
|
+
const res = await this._fetch("POST", "/entities", {
|
|
40
|
+
type,
|
|
41
|
+
attributes,
|
|
42
|
+
links
|
|
43
|
+
});
|
|
44
|
+
return res.id;
|
|
45
|
+
}
|
|
46
|
+
const pool = this._getPool();
|
|
47
|
+
const tenantId = this.opts.tenantId ?? null;
|
|
48
|
+
const kernel = pool.get(tenantId);
|
|
49
|
+
const entityId = `${type.toLowerCase()}:${crypto.randomUUID()}`;
|
|
50
|
+
await kernel.createEntity(entityId, type, attributes, links);
|
|
51
|
+
return entityId;
|
|
52
|
+
}
|
|
53
|
+
async read(id) {
|
|
54
|
+
if (isRemote(this.opts)) {
|
|
55
|
+
try {
|
|
56
|
+
return await this._fetch("GET", `/entities/${encodeURIComponent(id)}`);
|
|
57
|
+
} catch (err) {
|
|
58
|
+
if (err?.status === 404)
|
|
59
|
+
return null;
|
|
60
|
+
throw err;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
const pool = this._getPool();
|
|
64
|
+
const tenantId = this.opts.tenantId ?? null;
|
|
65
|
+
const kernel = pool.get(tenantId);
|
|
66
|
+
const entity = kernel.getEntity(id);
|
|
67
|
+
if (!entity)
|
|
68
|
+
return null;
|
|
69
|
+
return entityToPlain(entity);
|
|
70
|
+
}
|
|
71
|
+
async update(id, attributes) {
|
|
72
|
+
if (isRemote(this.opts)) {
|
|
73
|
+
await this._fetch("PUT", `/entities/${encodeURIComponent(id)}`, attributes);
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
const pool = this._getPool();
|
|
77
|
+
const tenantId = this.opts.tenantId ?? null;
|
|
78
|
+
const kernel = pool.get(tenantId);
|
|
79
|
+
await kernel.updateEntity(id, attributes);
|
|
80
|
+
}
|
|
81
|
+
async delete(id) {
|
|
82
|
+
if (isRemote(this.opts)) {
|
|
83
|
+
await this._fetch("DELETE", `/entities/${encodeURIComponent(id)}`);
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
const pool = this._getPool();
|
|
87
|
+
const tenantId = this.opts.tenantId ?? null;
|
|
88
|
+
const kernel = pool.get(tenantId);
|
|
89
|
+
await kernel.deleteEntity(id);
|
|
90
|
+
}
|
|
91
|
+
async list(type, opts = {}) {
|
|
92
|
+
if (isRemote(this.opts)) {
|
|
93
|
+
const params = new URLSearchParams;
|
|
94
|
+
if (type)
|
|
95
|
+
params.set("type", type);
|
|
96
|
+
if (opts.limit)
|
|
97
|
+
params.set("limit", String(opts.limit));
|
|
98
|
+
if (opts.offset)
|
|
99
|
+
params.set("offset", String(opts.offset));
|
|
100
|
+
const qs = params.toString() ? `?${params}` : "";
|
|
101
|
+
return await this._fetch("GET", `/entities${qs}`);
|
|
102
|
+
}
|
|
103
|
+
const pool = this._getPool();
|
|
104
|
+
const tenantId = this.opts.tenantId ?? null;
|
|
105
|
+
const kernel = pool.get(tenantId);
|
|
106
|
+
const entities = kernel.listEntities(type, opts.filters);
|
|
107
|
+
const limit = opts.limit ?? 100;
|
|
108
|
+
const offset = opts.offset ?? 0;
|
|
109
|
+
const page = entities.slice(offset, offset + limit);
|
|
110
|
+
return {
|
|
111
|
+
data: page.map(entityToPlain),
|
|
112
|
+
total: entities.length,
|
|
113
|
+
limit,
|
|
114
|
+
offset
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
async query(eql) {
|
|
118
|
+
if (isRemote(this.opts)) {
|
|
119
|
+
return await this._fetch("POST", "/query", {
|
|
120
|
+
query: eql
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
const { parseSimple } = await import("./index-2r4jxwnb.js");
|
|
124
|
+
const { QueryEngine } = await import("./engine-y0724kjq.js");
|
|
125
|
+
const pool = this._getPool();
|
|
126
|
+
const tenantId = this.opts.tenantId ?? null;
|
|
127
|
+
const kernel = pool.get(tenantId);
|
|
128
|
+
const parsed = parseSimple(eql);
|
|
129
|
+
const engine = new QueryEngine(kernel.getStore());
|
|
130
|
+
return engine.execute(parsed);
|
|
131
|
+
}
|
|
132
|
+
async upload(data, contentType = "application/octet-stream") {
|
|
133
|
+
const raw = data instanceof ArrayBuffer ? new Uint8Array(data) : data;
|
|
134
|
+
const cleanBuf = raw.buffer.slice(raw.byteOffset, raw.byteOffset + raw.byteLength);
|
|
135
|
+
const buffer = new Uint8Array(cleanBuf);
|
|
136
|
+
if (isRemote(this.opts)) {
|
|
137
|
+
const res = await fetch(`${this.opts.url}/upload`, {
|
|
138
|
+
method: "POST",
|
|
139
|
+
headers: {
|
|
140
|
+
"Content-Type": contentType,
|
|
141
|
+
...this.opts.apiKey ? { Authorization: `Bearer ${this.opts.apiKey}` } : {}
|
|
142
|
+
},
|
|
143
|
+
body: cleanBuf
|
|
144
|
+
});
|
|
145
|
+
if (!res.ok)
|
|
146
|
+
throw new FetchError(res.status, await res.text());
|
|
147
|
+
return await res.json();
|
|
148
|
+
}
|
|
149
|
+
const hashBuf = await crypto.subtle.digest("SHA-256", buffer);
|
|
150
|
+
const hash = `blob:${Array.from(new Uint8Array(hashBuf)).map((b) => b.toString(16).padStart(2, "0")).join("")}`;
|
|
151
|
+
const pool = this._getPool();
|
|
152
|
+
const tenantId = this.opts.tenantId ?? null;
|
|
153
|
+
const kernel = pool.get(tenantId);
|
|
154
|
+
const backend = kernel.getBackend();
|
|
155
|
+
if (!backend.hasBlob(hash))
|
|
156
|
+
backend.putBlob(hash, buffer);
|
|
157
|
+
return { hash, size: buffer.length, contentType };
|
|
158
|
+
}
|
|
159
|
+
async getFile(hash) {
|
|
160
|
+
if (isRemote(this.opts)) {
|
|
161
|
+
const res = await fetch(`${this.opts.url}/files/${encodeURIComponent(hash)}`, {
|
|
162
|
+
headers: this.opts.apiKey ? { Authorization: `Bearer ${this.opts.apiKey}` } : {}
|
|
163
|
+
});
|
|
164
|
+
if (res.status === 404)
|
|
165
|
+
return null;
|
|
166
|
+
if (!res.ok)
|
|
167
|
+
throw new FetchError(res.status, await res.text());
|
|
168
|
+
return new Uint8Array(await res.arrayBuffer());
|
|
169
|
+
}
|
|
170
|
+
const pool = this._getPool();
|
|
171
|
+
const tenantId = this.opts.tenantId ?? null;
|
|
172
|
+
const kernel = pool.get(tenantId);
|
|
173
|
+
const backend = kernel.getBackend();
|
|
174
|
+
return backend.getBlob(hash) ?? null;
|
|
175
|
+
}
|
|
176
|
+
async register(email, password, name) {
|
|
177
|
+
if (!isRemote(this.opts))
|
|
178
|
+
throw new Error("register() requires remote mode");
|
|
179
|
+
return await this._fetch("POST", "/auth/register", {
|
|
180
|
+
email,
|
|
181
|
+
password,
|
|
182
|
+
name
|
|
183
|
+
});
|
|
184
|
+
}
|
|
185
|
+
async login(email, password) {
|
|
186
|
+
if (!isRemote(this.opts))
|
|
187
|
+
throw new Error("login() requires remote mode");
|
|
188
|
+
return await this._fetch("POST", "/auth/login", {
|
|
189
|
+
email,
|
|
190
|
+
password
|
|
191
|
+
});
|
|
192
|
+
}
|
|
193
|
+
setToken(token) {
|
|
194
|
+
if (!isRemote(this.opts))
|
|
195
|
+
return;
|
|
196
|
+
this.opts.apiKey = token;
|
|
197
|
+
}
|
|
198
|
+
subscribe(eql, callback) {
|
|
199
|
+
if (!isRemote(this.opts)) {
|
|
200
|
+
throw new Error("subscribe() requires remote mode (connect to a running server)");
|
|
201
|
+
}
|
|
202
|
+
const subId = `sub_${crypto.randomUUID()}`;
|
|
203
|
+
this._subCallbacks.set(subId, callback);
|
|
204
|
+
this._ensureWs().then((ws) => {
|
|
205
|
+
ws.send(JSON.stringify({ type: "subscribe", id: subId, query: eql }));
|
|
206
|
+
});
|
|
207
|
+
return {
|
|
208
|
+
unsubscribe: () => {
|
|
209
|
+
this._subCallbacks.delete(subId);
|
|
210
|
+
this._ws?.send(JSON.stringify({ type: "unsubscribe", id: subId }));
|
|
211
|
+
}
|
|
212
|
+
};
|
|
213
|
+
}
|
|
214
|
+
disconnect() {
|
|
215
|
+
this._ws?.close();
|
|
216
|
+
this._ws = null;
|
|
217
|
+
}
|
|
218
|
+
close() {
|
|
219
|
+
this._pool?.closeAll();
|
|
220
|
+
this._pool = null;
|
|
221
|
+
this.disconnect();
|
|
222
|
+
}
|
|
223
|
+
_getPool() {
|
|
224
|
+
if (!this._pool) {
|
|
225
|
+
const path = this.opts.path;
|
|
226
|
+
this._pool = new TenantPool(path);
|
|
227
|
+
}
|
|
228
|
+
return this._pool;
|
|
229
|
+
}
|
|
230
|
+
async _fetch(method, path, body) {
|
|
231
|
+
const opts = this.opts;
|
|
232
|
+
const url = `${opts.url}${path}`;
|
|
233
|
+
const res = await fetch(url, {
|
|
234
|
+
method,
|
|
235
|
+
headers: {
|
|
236
|
+
"Content-Type": "application/json",
|
|
237
|
+
...opts.apiKey ? { Authorization: `Bearer ${opts.apiKey}` } : {}
|
|
238
|
+
},
|
|
239
|
+
body: body !== undefined ? JSON.stringify(body) : undefined
|
|
240
|
+
});
|
|
241
|
+
const data = await res.json();
|
|
242
|
+
if (!res.ok)
|
|
243
|
+
throw new FetchError(res.status, data?.message ?? res.statusText, data);
|
|
244
|
+
return data;
|
|
245
|
+
}
|
|
246
|
+
async _ensureWs() {
|
|
247
|
+
if (this._ws && this._ws.readyState === WebSocket.OPEN)
|
|
248
|
+
return this._ws;
|
|
249
|
+
const opts = this.opts;
|
|
250
|
+
const wsUrl = opts.url.replace(/^https?/, opts.url.startsWith("https") ? "wss" : "ws") + "/realtime";
|
|
251
|
+
return new Promise((resolve, reject) => {
|
|
252
|
+
const ws = new WebSocket(wsUrl);
|
|
253
|
+
ws.onopen = () => {
|
|
254
|
+
this._ws = ws;
|
|
255
|
+
resolve(ws);
|
|
256
|
+
};
|
|
257
|
+
ws.onerror = (e) => reject(e);
|
|
258
|
+
ws.onmessage = (e) => {
|
|
259
|
+
try {
|
|
260
|
+
const msg = JSON.parse(e.data);
|
|
261
|
+
if (msg.type === "data" && this._subCallbacks.has(msg.id)) {
|
|
262
|
+
this._subCallbacks.get(msg.id)(msg.result, msg.diff);
|
|
263
|
+
}
|
|
264
|
+
} catch {}
|
|
265
|
+
};
|
|
266
|
+
ws.onclose = () => {
|
|
267
|
+
this._ws = null;
|
|
268
|
+
};
|
|
269
|
+
});
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
function entityToPlain(entity) {
|
|
273
|
+
const obj = { id: entity.id, type: entity.type };
|
|
274
|
+
for (const f of entity.facts) {
|
|
275
|
+
if (f.a !== "type")
|
|
276
|
+
obj[f.a] = f.v;
|
|
277
|
+
}
|
|
278
|
+
return obj;
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
class FetchError extends Error {
|
|
282
|
+
status;
|
|
283
|
+
body;
|
|
284
|
+
constructor(status, message, body) {
|
|
285
|
+
super(`HTTP ${status}: ${message}`);
|
|
286
|
+
this.status = status;
|
|
287
|
+
this.body = body;
|
|
288
|
+
this.name = "FetchError";
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
export { TrellisDb, FetchError };
|
|
@@ -0,0 +1,326 @@
|
|
|
1
|
+
// @bun
|
|
2
|
+
import {
|
|
3
|
+
__esm
|
|
4
|
+
} from "./index-a76rekgs.js";
|
|
5
|
+
|
|
6
|
+
// src/core/store/eav-store.ts
|
|
7
|
+
function* flatten(obj, base = "") {
|
|
8
|
+
if (Array.isArray(obj)) {
|
|
9
|
+
for (const v of obj) {
|
|
10
|
+
yield* flatten(v, base);
|
|
11
|
+
}
|
|
12
|
+
} else if (obj && typeof obj === "object") {
|
|
13
|
+
for (const [k, v] of Object.entries(obj)) {
|
|
14
|
+
yield* flatten(v, base ? `${base}.${k}` : k);
|
|
15
|
+
}
|
|
16
|
+
} else {
|
|
17
|
+
yield [base, obj];
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
function jsonEntityFacts(entityId, root, type) {
|
|
21
|
+
const facts = [{ e: entityId, a: "type", v: type }];
|
|
22
|
+
for (const [a, v] of flatten(root)) {
|
|
23
|
+
if (v === undefined || v === null)
|
|
24
|
+
continue;
|
|
25
|
+
if (Array.isArray(v)) {
|
|
26
|
+
for (const el of v) {
|
|
27
|
+
facts.push({ e: entityId, a, v: el });
|
|
28
|
+
}
|
|
29
|
+
} else if (typeof v === "object") {} else {
|
|
30
|
+
facts.push({ e: entityId, a, v });
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
return facts;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
class EAVStore {
|
|
37
|
+
facts = [];
|
|
38
|
+
links = [];
|
|
39
|
+
catalog = new Map;
|
|
40
|
+
eavIndex = new Map;
|
|
41
|
+
aevIndex = new Map;
|
|
42
|
+
aveIndex = new Map;
|
|
43
|
+
linkIndex = new Map;
|
|
44
|
+
linkReverseIndex = new Map;
|
|
45
|
+
linkAttrIndex = new Map;
|
|
46
|
+
distinct = new Map;
|
|
47
|
+
addFacts(facts) {
|
|
48
|
+
for (let i = 0;i < facts.length; i++) {
|
|
49
|
+
const fact = facts[i];
|
|
50
|
+
if (fact) {
|
|
51
|
+
this.facts.push(fact);
|
|
52
|
+
this.updateIndexes(fact, this.facts.length - 1);
|
|
53
|
+
this.updateCatalog(fact);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
addLinks(links) {
|
|
58
|
+
for (const link of links) {
|
|
59
|
+
this.links.push(link);
|
|
60
|
+
this.updateLinkIndexes(link);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
deleteFacts(factsToDelete) {
|
|
64
|
+
for (const fact of factsToDelete) {
|
|
65
|
+
const valueKey = this.valueKey(fact.v);
|
|
66
|
+
const indices = this.aveIndex.get(fact.a)?.get(valueKey);
|
|
67
|
+
if (!indices)
|
|
68
|
+
continue;
|
|
69
|
+
let foundIdx = -1;
|
|
70
|
+
for (const idx of indices) {
|
|
71
|
+
const storedFact = this.facts[idx];
|
|
72
|
+
if (storedFact && storedFact.e === fact.e && storedFact.a === fact.a) {
|
|
73
|
+
foundIdx = idx;
|
|
74
|
+
break;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
if (foundIdx !== -1) {
|
|
78
|
+
this.facts[foundIdx] = undefined;
|
|
79
|
+
this.eavIndex.get(fact.e)?.get(fact.a)?.delete(foundIdx);
|
|
80
|
+
this.aevIndex.get(fact.a)?.get(fact.e)?.delete(foundIdx);
|
|
81
|
+
this.aveIndex.get(fact.a)?.get(valueKey)?.delete(foundIdx);
|
|
82
|
+
const entry = this.catalog.get(fact.a);
|
|
83
|
+
if (entry) {}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
deleteLinks(linksToDelete) {
|
|
88
|
+
for (const link of linksToDelete) {
|
|
89
|
+
const initialLen = this.links.length;
|
|
90
|
+
this.links = this.links.filter((l) => !(l.e1 === link.e1 && l.a === link.a && l.e2 === link.e2));
|
|
91
|
+
if (this.links.length < initialLen) {
|
|
92
|
+
this.linkIndex.get(link.e1)?.get(link.a)?.delete(link.e2);
|
|
93
|
+
this.linkReverseIndex.get(link.e2)?.get(link.a)?.delete(link.e1);
|
|
94
|
+
const attrPairs = this.linkAttrIndex.get(link.a);
|
|
95
|
+
if (attrPairs) {
|
|
96
|
+
for (const pair of attrPairs) {
|
|
97
|
+
if (pair[0] === link.e1 && pair[1] === link.e2) {
|
|
98
|
+
attrPairs.delete(pair);
|
|
99
|
+
break;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
updateIndexes(fact, index) {
|
|
107
|
+
if (!this.eavIndex.has(fact.e)) {
|
|
108
|
+
this.eavIndex.set(fact.e, new Map);
|
|
109
|
+
}
|
|
110
|
+
if (!this.eavIndex.get(fact.e).has(fact.a)) {
|
|
111
|
+
this.eavIndex.get(fact.e).set(fact.a, new Set);
|
|
112
|
+
}
|
|
113
|
+
this.eavIndex.get(fact.e).get(fact.a).add(index);
|
|
114
|
+
if (!this.aevIndex.has(fact.a)) {
|
|
115
|
+
this.aevIndex.set(fact.a, new Map);
|
|
116
|
+
}
|
|
117
|
+
if (!this.aevIndex.get(fact.a).has(fact.e)) {
|
|
118
|
+
this.aevIndex.get(fact.a).set(fact.e, new Set);
|
|
119
|
+
}
|
|
120
|
+
this.aevIndex.get(fact.a).get(fact.e).add(index);
|
|
121
|
+
if (!this.aveIndex.has(fact.a)) {
|
|
122
|
+
this.aveIndex.set(fact.a, new Map);
|
|
123
|
+
}
|
|
124
|
+
const valueKey = this.valueKey(fact.v);
|
|
125
|
+
if (!this.aveIndex.get(fact.a).has(valueKey)) {
|
|
126
|
+
this.aveIndex.get(fact.a).set(valueKey, new Set);
|
|
127
|
+
}
|
|
128
|
+
this.aveIndex.get(fact.a).get(valueKey).add(index);
|
|
129
|
+
}
|
|
130
|
+
updateLinkIndexes(link) {
|
|
131
|
+
if (!this.linkIndex.has(link.e1)) {
|
|
132
|
+
this.linkIndex.set(link.e1, new Map);
|
|
133
|
+
}
|
|
134
|
+
const e1Attrs = this.linkIndex.get(link.e1);
|
|
135
|
+
if (!e1Attrs.has(link.a)) {
|
|
136
|
+
e1Attrs.set(link.a, new Set);
|
|
137
|
+
}
|
|
138
|
+
e1Attrs.get(link.a).add(link.e2);
|
|
139
|
+
if (!this.linkReverseIndex.has(link.e2)) {
|
|
140
|
+
this.linkReverseIndex.set(link.e2, new Map);
|
|
141
|
+
}
|
|
142
|
+
const e2Attrs = this.linkReverseIndex.get(link.e2);
|
|
143
|
+
if (!e2Attrs.has(link.a)) {
|
|
144
|
+
e2Attrs.set(link.a, new Set);
|
|
145
|
+
}
|
|
146
|
+
e2Attrs.get(link.a).add(link.e1);
|
|
147
|
+
if (!this.linkAttrIndex.has(link.a)) {
|
|
148
|
+
this.linkAttrIndex.set(link.a, new Set);
|
|
149
|
+
}
|
|
150
|
+
this.linkAttrIndex.get(link.a).add([link.e1, link.e2]);
|
|
151
|
+
}
|
|
152
|
+
valueKey(v) {
|
|
153
|
+
if (v instanceof Date)
|
|
154
|
+
return `date:${v.toISOString()}`;
|
|
155
|
+
return `${typeof v}:${v}`;
|
|
156
|
+
}
|
|
157
|
+
updateCatalog(fact) {
|
|
158
|
+
const entry = this.catalog.get(fact.a) || {
|
|
159
|
+
attribute: fact.a,
|
|
160
|
+
type: this.inferType(fact.v),
|
|
161
|
+
cardinality: "one",
|
|
162
|
+
distinctCount: 0,
|
|
163
|
+
examples: []
|
|
164
|
+
};
|
|
165
|
+
const factType = this.inferType(fact.v);
|
|
166
|
+
if (entry.type !== factType && entry.type !== "mixed") {
|
|
167
|
+
entry.type = "mixed";
|
|
168
|
+
}
|
|
169
|
+
const entityAttrs = this.eavIndex.get(fact.e)?.get(fact.a);
|
|
170
|
+
if (entityAttrs && entityAttrs.size > 1) {
|
|
171
|
+
entry.cardinality = "many";
|
|
172
|
+
}
|
|
173
|
+
const k = this.valueKey(fact.v);
|
|
174
|
+
const s = this.distinct.get(fact.a) || (this.distinct.set(fact.a, new Set), this.distinct.get(fact.a));
|
|
175
|
+
s.add(k);
|
|
176
|
+
entry.distinctCount = s.size;
|
|
177
|
+
if (entry.examples.length < 5 && !entry.examples.includes(fact.v)) {
|
|
178
|
+
entry.examples.push(fact.v);
|
|
179
|
+
}
|
|
180
|
+
if (typeof fact.v === "number") {
|
|
181
|
+
entry.min = Math.min(entry.min ?? fact.v, fact.v);
|
|
182
|
+
entry.max = Math.max(entry.max ?? fact.v, fact.v);
|
|
183
|
+
}
|
|
184
|
+
this.catalog.set(fact.a, entry);
|
|
185
|
+
}
|
|
186
|
+
inferType(v) {
|
|
187
|
+
if (typeof v === "string")
|
|
188
|
+
return "string";
|
|
189
|
+
if (typeof v === "number")
|
|
190
|
+
return "number";
|
|
191
|
+
if (typeof v === "boolean")
|
|
192
|
+
return "boolean";
|
|
193
|
+
if (v instanceof Date)
|
|
194
|
+
return "date";
|
|
195
|
+
return "mixed";
|
|
196
|
+
}
|
|
197
|
+
getFactsByEntity(entity) {
|
|
198
|
+
const indices = this.eavIndex.get(entity);
|
|
199
|
+
if (!indices)
|
|
200
|
+
return [];
|
|
201
|
+
const result = [];
|
|
202
|
+
for (const attrIndices of indices.values()) {
|
|
203
|
+
for (const idx of attrIndices) {
|
|
204
|
+
const fact = this.facts[idx];
|
|
205
|
+
if (fact) {
|
|
206
|
+
result.push(fact);
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
return result;
|
|
211
|
+
}
|
|
212
|
+
getFactsByAttribute(attribute) {
|
|
213
|
+
const indices = this.aevIndex.get(attribute);
|
|
214
|
+
if (!indices)
|
|
215
|
+
return [];
|
|
216
|
+
const result = [];
|
|
217
|
+
for (const entityIndices of indices.values()) {
|
|
218
|
+
for (const idx of entityIndices) {
|
|
219
|
+
const fact = this.facts[idx];
|
|
220
|
+
if (fact) {
|
|
221
|
+
result.push(fact);
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
return result;
|
|
226
|
+
}
|
|
227
|
+
getFactsByValue(attribute, value) {
|
|
228
|
+
const indices = this.aveIndex.get(attribute)?.get(this.valueKey(value));
|
|
229
|
+
if (!indices)
|
|
230
|
+
return [];
|
|
231
|
+
return Array.from(indices).map((idx) => this.facts[idx]).filter((fact) => fact !== undefined);
|
|
232
|
+
}
|
|
233
|
+
getCatalog() {
|
|
234
|
+
return Array.from(this.catalog.values());
|
|
235
|
+
}
|
|
236
|
+
getCatalogEntry(attribute) {
|
|
237
|
+
return this.catalog.get(attribute);
|
|
238
|
+
}
|
|
239
|
+
getAllFacts() {
|
|
240
|
+
return this.facts.filter((f) => f !== undefined);
|
|
241
|
+
}
|
|
242
|
+
getAllLinks() {
|
|
243
|
+
return [...this.links];
|
|
244
|
+
}
|
|
245
|
+
getLinksByEntity(entity) {
|
|
246
|
+
const results = [];
|
|
247
|
+
const forwardLinks = this.linkIndex.get(entity);
|
|
248
|
+
if (forwardLinks) {
|
|
249
|
+
for (const [attr, targets] of forwardLinks) {
|
|
250
|
+
for (const target of targets) {
|
|
251
|
+
results.push({ e1: entity, a: attr, e2: target });
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
const reverseLinks = this.linkReverseIndex.get(entity);
|
|
256
|
+
if (reverseLinks) {
|
|
257
|
+
for (const [attr, sources] of reverseLinks) {
|
|
258
|
+
for (const source of sources) {
|
|
259
|
+
results.push({ e1: source, a: attr, e2: entity });
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
return results;
|
|
264
|
+
}
|
|
265
|
+
getLinksByAttribute(attribute) {
|
|
266
|
+
const results = [];
|
|
267
|
+
const links = this.linkAttrIndex.get(attribute);
|
|
268
|
+
if (links) {
|
|
269
|
+
for (const [e1, e2] of links) {
|
|
270
|
+
results.push({ e1, a: attribute, e2 });
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
return results;
|
|
274
|
+
}
|
|
275
|
+
getLinksByEntityAndAttribute(entity, attribute) {
|
|
276
|
+
const results = [];
|
|
277
|
+
const attrs = this.linkIndex.get(entity);
|
|
278
|
+
if (attrs) {
|
|
279
|
+
const targets = attrs.get(attribute);
|
|
280
|
+
if (targets) {
|
|
281
|
+
for (const target of targets) {
|
|
282
|
+
results.push({ e1: entity, a: attribute, e2: target });
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
return results;
|
|
287
|
+
}
|
|
288
|
+
getStats() {
|
|
289
|
+
return {
|
|
290
|
+
totalFacts: this.facts.length,
|
|
291
|
+
totalLinks: this.links.length,
|
|
292
|
+
uniqueEntities: this.eavIndex.size,
|
|
293
|
+
uniqueAttributes: this.aevIndex.size,
|
|
294
|
+
catalogEntries: this.catalog.size
|
|
295
|
+
};
|
|
296
|
+
}
|
|
297
|
+
snapshot() {
|
|
298
|
+
return {
|
|
299
|
+
facts: this.facts.filter((f) => f !== undefined),
|
|
300
|
+
links: [...this.links],
|
|
301
|
+
catalog: this.getCatalog()
|
|
302
|
+
};
|
|
303
|
+
}
|
|
304
|
+
restore(snapshot) {
|
|
305
|
+
this.facts = [];
|
|
306
|
+
this.links = [];
|
|
307
|
+
this.catalog.clear();
|
|
308
|
+
this.eavIndex.clear();
|
|
309
|
+
this.aevIndex.clear();
|
|
310
|
+
this.aveIndex.clear();
|
|
311
|
+
this.linkIndex.clear();
|
|
312
|
+
this.linkReverseIndex.clear();
|
|
313
|
+
this.linkAttrIndex.clear();
|
|
314
|
+
this.distinct.clear();
|
|
315
|
+
this.addFacts(snapshot.facts);
|
|
316
|
+
this.addLinks(snapshot.links);
|
|
317
|
+
if (snapshot.catalog) {
|
|
318
|
+
for (const entry of snapshot.catalog) {
|
|
319
|
+
this.catalog.set(entry.attribute, entry);
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
var init_eav_store = () => {};
|
|
325
|
+
|
|
326
|
+
export { flatten, jsonEntityFacts, EAVStore, init_eav_store };
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
// @bun
|
|
2
|
+
import {
|
|
3
|
+
SqliteKernelBackend,
|
|
4
|
+
TrellisKernel
|
|
5
|
+
} from "./index-7t92ej34.js";
|
|
6
|
+
|
|
7
|
+
// src/server/tenancy.ts
|
|
8
|
+
import { existsSync, mkdirSync } from "fs";
|
|
9
|
+
import { join, resolve } from "path";
|
|
10
|
+
var DEFAULT_TENANT = "default";
|
|
11
|
+
var TENANTS_DIR = "tenants";
|
|
12
|
+
|
|
13
|
+
class TenantPool {
|
|
14
|
+
pool = new Map;
|
|
15
|
+
dbPath;
|
|
16
|
+
agentId;
|
|
17
|
+
constructor(dbPath, agentId = "trellis-db") {
|
|
18
|
+
this.dbPath = resolve(dbPath);
|
|
19
|
+
this.agentId = agentId;
|
|
20
|
+
this._ensureDirs();
|
|
21
|
+
}
|
|
22
|
+
get(tenantId) {
|
|
23
|
+
const id = tenantId ?? DEFAULT_TENANT;
|
|
24
|
+
if (!this.pool.has(id)) {
|
|
25
|
+
this.pool.set(id, this._createKernel(id));
|
|
26
|
+
}
|
|
27
|
+
return this.pool.get(id);
|
|
28
|
+
}
|
|
29
|
+
has(tenantId) {
|
|
30
|
+
return this.pool.has(tenantId);
|
|
31
|
+
}
|
|
32
|
+
activeTenants() {
|
|
33
|
+
return Array.from(this.pool.keys());
|
|
34
|
+
}
|
|
35
|
+
close(tenantId) {
|
|
36
|
+
const kernel = this.pool.get(tenantId);
|
|
37
|
+
if (kernel) {
|
|
38
|
+
kernel.close();
|
|
39
|
+
this.pool.delete(tenantId);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
closeAll() {
|
|
43
|
+
for (const kernel of this.pool.values()) {
|
|
44
|
+
kernel.close();
|
|
45
|
+
}
|
|
46
|
+
this.pool.clear();
|
|
47
|
+
}
|
|
48
|
+
dbFilePath(tenantId) {
|
|
49
|
+
const id = tenantId ?? DEFAULT_TENANT;
|
|
50
|
+
if (id === DEFAULT_TENANT) {
|
|
51
|
+
return join(this.dbPath, "default.sqlite");
|
|
52
|
+
}
|
|
53
|
+
return join(this.dbPath, TENANTS_DIR, `${id}.sqlite`);
|
|
54
|
+
}
|
|
55
|
+
_createKernel(tenantId) {
|
|
56
|
+
const sqlitePath = this.dbFilePath(tenantId);
|
|
57
|
+
const backend = new SqliteKernelBackend(sqlitePath);
|
|
58
|
+
const kernel = new TrellisKernel({
|
|
59
|
+
backend,
|
|
60
|
+
agentId: this.agentId,
|
|
61
|
+
snapshotThreshold: 1000
|
|
62
|
+
});
|
|
63
|
+
kernel.boot();
|
|
64
|
+
return kernel;
|
|
65
|
+
}
|
|
66
|
+
_ensureDirs() {
|
|
67
|
+
if (!existsSync(this.dbPath)) {
|
|
68
|
+
mkdirSync(this.dbPath, { recursive: true });
|
|
69
|
+
}
|
|
70
|
+
const tenantsPath = join(this.dbPath, TENANTS_DIR);
|
|
71
|
+
if (!existsSync(tenantsPath)) {
|
|
72
|
+
mkdirSync(tenantsPath, { recursive: true });
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
export { DEFAULT_TENANT, TenantPool };
|