@treeseed/sdk 0.10.28 → 0.11.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (148) hide show
  1. package/README.md +207 -6
  2. package/dist/capacity-provider.d.ts +3 -1
  3. package/dist/capacity-provider.js +25 -5
  4. package/dist/control-plane.d.ts +1 -0
  5. package/dist/control-plane.js +38 -13
  6. package/dist/db/market-schema.d.ts +8860 -6172
  7. package/dist/db/market-schema.js +108 -0
  8. package/dist/db/node-sqlite.js +7 -2
  9. package/dist/hosting/apps.d.ts +12 -0
  10. package/dist/hosting/apps.js +107 -0
  11. package/dist/hosting/builtins.d.ts +25 -0
  12. package/dist/hosting/builtins.js +791 -0
  13. package/dist/hosting/contracts.d.ts +207 -0
  14. package/dist/hosting/contracts.js +0 -0
  15. package/dist/hosting/graph.d.ts +192 -0
  16. package/dist/hosting/graph.js +1106 -0
  17. package/dist/hosting/index.d.ts +4 -0
  18. package/dist/hosting/index.js +4 -0
  19. package/dist/index.d.ts +10 -3
  20. package/dist/index.js +63 -6
  21. package/dist/managed-dependencies.js +1 -2
  22. package/dist/market-client.d.ts +63 -3
  23. package/dist/market-client.js +83 -11
  24. package/dist/operations/services/bootstrap-runner.d.ts +3 -1
  25. package/dist/operations/services/bootstrap-runner.js +22 -2
  26. package/dist/operations/services/config-runtime.d.ts +10 -5
  27. package/dist/operations/services/config-runtime.js +209 -66
  28. package/dist/operations/services/deploy.d.ts +70 -7
  29. package/dist/operations/services/deploy.js +579 -64
  30. package/dist/operations/services/deployment-readiness.d.ts +30 -0
  31. package/dist/operations/services/deployment-readiness.js +175 -0
  32. package/dist/operations/services/git-workflow.d.ts +2 -1
  33. package/dist/operations/services/git-workflow.js +9 -3
  34. package/dist/operations/services/github-actions-verification.d.ts +1 -0
  35. package/dist/operations/services/github-actions-verification.js +1 -0
  36. package/dist/operations/services/github-api.js +1 -1
  37. package/dist/operations/services/github-automation.d.ts +1 -1
  38. package/dist/operations/services/github-automation.js +4 -3
  39. package/dist/operations/services/github-credentials.d.ts +13 -0
  40. package/dist/operations/services/github-credentials.js +58 -0
  41. package/dist/operations/services/hosted-service-checks.d.ts +63 -0
  42. package/dist/operations/services/hosted-service-checks.js +327 -0
  43. package/dist/operations/services/hub-provider-launch.js +3 -3
  44. package/dist/operations/services/live-hosted-service-checks.d.ts +25 -0
  45. package/dist/operations/services/live-hosted-service-checks.js +350 -0
  46. package/dist/operations/services/managed-host-security.js +1 -1
  47. package/dist/operations/services/operations-runner-smoke.d.ts +30 -0
  48. package/dist/operations/services/operations-runner-smoke.js +180 -0
  49. package/dist/operations/services/package-adapters.d.ts +95 -0
  50. package/dist/operations/services/package-adapters.js +288 -0
  51. package/dist/operations/services/package-reference-policy.d.ts +1 -0
  52. package/dist/operations/services/package-reference-policy.js +15 -2
  53. package/dist/operations/services/project-platform.d.ts +80 -22
  54. package/dist/operations/services/project-platform.js +49 -8
  55. package/dist/operations/services/project-web-monitor.js +26 -4
  56. package/dist/operations/services/railway-api.d.ts +88 -5
  57. package/dist/operations/services/railway-api.js +626 -35
  58. package/dist/operations/services/railway-deploy.d.ts +46 -40
  59. package/dist/operations/services/railway-deploy.js +261 -293
  60. package/dist/operations/services/release-candidate.d.ts +19 -0
  61. package/dist/operations/services/release-candidate.js +375 -38
  62. package/dist/operations/services/repository-save-orchestrator.d.ts +3 -1
  63. package/dist/operations/services/repository-save-orchestrator.js +279 -66
  64. package/dist/operations/services/runtime-tools.d.ts +1 -0
  65. package/dist/operations/services/runtime-tools.js +10 -9
  66. package/dist/operations/services/verification-cache.d.ts +25 -0
  67. package/dist/operations/services/verification-cache.js +71 -0
  68. package/dist/operations/services/workspace-dependency-mode.js +9 -1
  69. package/dist/operations/services/workspace-save.js +1 -1
  70. package/dist/operations/services/workspace-tools.js +2 -1
  71. package/dist/platform/contracts.d.ts +32 -1
  72. package/dist/platform/deploy-config.js +73 -8
  73. package/dist/platform/env.yaml +163 -35
  74. package/dist/platform/environment.d.ts +1 -0
  75. package/dist/platform/environment.js +74 -5
  76. package/dist/platform/plugin.d.ts +9 -0
  77. package/dist/platform-operation-store.js +2 -2
  78. package/dist/platform-operations.js +1 -1
  79. package/dist/reconcile/bootstrap-systems.js +2 -2
  80. package/dist/reconcile/builtin-adapters.js +372 -189
  81. package/dist/reconcile/contracts.d.ts +9 -5
  82. package/dist/reconcile/desired-state.d.ts +1 -0
  83. package/dist/reconcile/desired-state.js +5 -5
  84. package/dist/reconcile/engine.d.ts +5 -2
  85. package/dist/reconcile/engine.js +53 -32
  86. package/dist/reconcile/index.d.ts +2 -0
  87. package/dist/reconcile/index.js +2 -0
  88. package/dist/reconcile/live-acceptance.d.ts +79 -0
  89. package/dist/reconcile/live-acceptance.js +1615 -0
  90. package/dist/reconcile/platform.d.ts +104 -0
  91. package/dist/reconcile/platform.js +100 -0
  92. package/dist/reconcile/state.js +4 -4
  93. package/dist/reconcile/units.js +2 -2
  94. package/dist/scripts/deployment-readiness.js +20 -0
  95. package/dist/scripts/generate-treedx-openapi-types.js +186 -0
  96. package/dist/scripts/operations-runner-smoke.js +16 -0
  97. package/dist/scripts/release-verify.js +4 -1
  98. package/dist/scripts/tenant-workflow-action.js +10 -1
  99. package/dist/sdk-types.d.ts +169 -4
  100. package/dist/sdk-types.js +20 -2
  101. package/dist/sdk.d.ts +35 -24
  102. package/dist/sdk.js +186 -17
  103. package/dist/template-launch-requirements.js +9 -0
  104. package/dist/treedx/adapters.d.ts +6 -0
  105. package/dist/treedx/adapters.js +36 -0
  106. package/dist/treedx/client.d.ts +222 -0
  107. package/dist/treedx/client.js +871 -0
  108. package/dist/treedx/errors.d.ts +13 -0
  109. package/dist/treedx/errors.js +17 -0
  110. package/dist/treedx/federated-client.d.ts +27 -0
  111. package/dist/treedx/federated-client.js +158 -0
  112. package/dist/treedx/generated/openapi-types.d.ts +3558 -0
  113. package/dist/treedx/generated/openapi-types.js +0 -0
  114. package/dist/treedx/graph-adapter.d.ts +33 -0
  115. package/dist/treedx/graph-adapter.js +156 -0
  116. package/dist/treedx/index.d.ts +14 -0
  117. package/dist/treedx/index.js +48 -0
  118. package/dist/treedx/market-integration.d.ts +27 -0
  119. package/dist/treedx/market-integration.js +131 -0
  120. package/dist/treedx/ports.d.ts +166 -0
  121. package/dist/treedx/ports.js +231 -0
  122. package/dist/treedx/query-adapter.d.ts +19 -0
  123. package/dist/treedx/query-adapter.js +62 -0
  124. package/dist/treedx/registry-client.d.ts +11 -0
  125. package/dist/treedx/registry-client.js +19 -0
  126. package/dist/treedx/repository-adapter.d.ts +45 -0
  127. package/dist/treedx/repository-adapter.js +308 -0
  128. package/dist/treedx/sdk-integration.d.ts +27 -0
  129. package/dist/treedx/sdk-integration.js +63 -0
  130. package/dist/treedx/types.d.ts +1084 -0
  131. package/dist/treedx/types.js +8 -0
  132. package/dist/treedx/workspace-adapter.d.ts +27 -0
  133. package/dist/treedx/workspace-adapter.js +65 -0
  134. package/dist/treedx-backends.d.ts +218 -0
  135. package/dist/treedx-backends.js +632 -0
  136. package/dist/treedx-client.d.ts +86 -0
  137. package/dist/treedx-client.js +175 -0
  138. package/dist/treeseed/template-catalog/catalog.fixture.json +23 -23
  139. package/dist/workflow/operations.d.ts +119 -13
  140. package/dist/workflow/operations.js +309 -53
  141. package/dist/workflow-state.d.ts +13 -0
  142. package/dist/workflow-state.js +43 -26
  143. package/dist/workflow-support.d.ts +11 -3
  144. package/dist/workflow-support.js +67 -3
  145. package/dist/workflow.d.ts +5 -0
  146. package/drizzle/market/0004_treedx_market_integration.sql +99 -0
  147. package/package.json +34 -3
  148. package/templates/github/deploy-web.workflow.yml +39 -6
@@ -0,0 +1,871 @@
1
+ import { TreeDxApiError } from "./errors.js";
2
+ function normalizeBaseUrl(baseUrl) {
3
+ return baseUrl.replace(/\/+$/u, "");
4
+ }
5
+ function isRecord(value) {
6
+ return typeof value === "object" && value !== null && !Array.isArray(value);
7
+ }
8
+ function isAbortError(error) {
9
+ return error instanceof DOMException && error.name === "AbortError" || error instanceof Error && error.name === "AbortError";
10
+ }
11
+ function stripOk(payload) {
12
+ if (!isRecord(payload)) {
13
+ return payload;
14
+ }
15
+ const { ok: _ok, ...rest } = payload;
16
+ return rest;
17
+ }
18
+ function firstPayload(payload, keys) {
19
+ if (isRecord(payload)) {
20
+ for (const key of keys) {
21
+ if (key in payload) {
22
+ return payload[key];
23
+ }
24
+ }
25
+ }
26
+ return stripOk(payload);
27
+ }
28
+ const TREEDX_CLIENT_OPERATION_MAP = {
29
+ health: "getHealth",
30
+ ready: "getReadiness",
31
+ deepHealth: "getDeepHealth",
32
+ adminDeepHealth: "getAdminDeepHealth",
33
+ metrics: "getMetrics",
34
+ prometheusMetrics: "getPrometheusMetrics",
35
+ whoami: "getWhoami",
36
+ authMode: "getAuthMode",
37
+ effectiveScope: "getEffectiveScope",
38
+ listCapabilities: "listCapabilities",
39
+ listCapabilityGrants: "listCapabilityGrants",
40
+ putCapabilityGrant: "putCapabilityGrant",
41
+ listAuditEvents: "listAuditEvents",
42
+ planFederatedQuery: "planFederationQuery",
43
+ buildSnapshot: "buildSnapshot",
44
+ getSnapshot: "getSnapshot",
45
+ exportArtifact: "exportArtifact",
46
+ syncMirror: "syncMirror",
47
+ fetchRemote: "syncRepository",
48
+ push: "pushRepository",
49
+ checkMirrorHealth: "checkMirrorHealth",
50
+ promoteMirror: "promoteMirror",
51
+ compactStorage: "compactAdminStorage",
52
+ backupStorage: "backupAdminStorage",
53
+ listStorageMigrations: "listStorageMigrations",
54
+ planStorageMigration: "planStorageMigration",
55
+ applyStorageMigration: "applyStorageMigration",
56
+ rollbackStorageMigration: "rollbackStorageMigration",
57
+ verifyStorageRestore: "verifyStorageRestore",
58
+ restoreStorage: "restoreStorage",
59
+ createMigration: "createMigration",
60
+ getMigration: "getMigration",
61
+ getNode: "getLocalNode",
62
+ listNodes: "listRegistryNodes",
63
+ getPlacement: "getRepositoryPlacement",
64
+ getRepository: "getRepository",
65
+ createWorkspace: "createWorkspace",
66
+ closeWorkspace: "closeWorkspace",
67
+ listTree: "listWorkspaceTree",
68
+ readFile: "readWorkspaceFile",
69
+ writeFile: "writeWorkspaceFile",
70
+ patchFile: "patchWorkspaceFile",
71
+ deleteFile: "deleteWorkspaceFile",
72
+ readBlob: "readRepositoryBlob",
73
+ writeBlob: "writeWorkspaceBlob",
74
+ deleteBlob: "deleteWorkspaceBlob",
75
+ downloadBlob: "downloadWorkspaceBlob",
76
+ uploadBlob: "uploadWorkspaceBlob",
77
+ createBlobUpload: "createWorkspaceBlobUpload",
78
+ uploadBlobPart: "uploadWorkspaceBlobPart",
79
+ completeBlobUpload: "completeWorkspaceBlobUpload",
80
+ abortBlobUpload: "abortWorkspaceBlobUpload",
81
+ listArtifacts: "listArtifacts",
82
+ getArtifact: "getArtifact",
83
+ deleteArtifact: "deleteArtifact",
84
+ cleanupArtifacts: "cleanupArtifacts",
85
+ search: "searchWorkspace",
86
+ status: "getWorkspaceStatus",
87
+ diff: "getWorkspaceDiff",
88
+ commit: "commitWorkspace",
89
+ exec: "execWorkspace",
90
+ readRepositoryFiles: "readRepositoryFile",
91
+ readRepositoryFile: "readRepositoryFile",
92
+ listRepositoryPaths: "listRepositoryPaths",
93
+ searchRepositoryFiles: "searchRepositoryFiles",
94
+ queryRepository: "queryRepository",
95
+ federatedSearch: "federatedSearch",
96
+ federatedQuery: "federatedQuery",
97
+ federatedContext: "federatedContextBuild",
98
+ federatedGraph: "federatedGraphQuery",
99
+ refreshGraph: "refreshRepositoryGraph",
100
+ getGraphRefreshJob: "getGraphRefreshJob",
101
+ refreshSearchIndex: "refreshSearchIndex",
102
+ getSearchIndexStatus: "getSearchIndexStatus",
103
+ compactSearchIndex: "compactSearchIndex",
104
+ queryGraph: "queryRepositoryGraph",
105
+ searchGraphFiles: "searchGraphFiles",
106
+ searchGraphSections: "searchGraphSections",
107
+ searchGraphEntities: "searchGraphEntities",
108
+ getGraphNode: "getGraphNode",
109
+ getRelated: "getRelatedGraphNodes",
110
+ getSubgraph: "getGraphSubgraph",
111
+ buildContext: "buildContext",
112
+ parseContextDsl: "parseContextQuery"
113
+ };
114
+ class TreeDxClient {
115
+ constructor(options) {
116
+ this.options = options;
117
+ this.baseUrl = normalizeBaseUrl(options.baseUrl);
118
+ this.token = options.token;
119
+ this.defaultRepoId = options.repoId;
120
+ this.fetchImpl = options.fetch ?? fetch;
121
+ }
122
+ options;
123
+ baseUrl;
124
+ token;
125
+ defaultRepoId;
126
+ fetchImpl;
127
+ repoId(inputRepoId) {
128
+ const repoId = inputRepoId ?? this.defaultRepoId;
129
+ if (!repoId) {
130
+ throw new TreeDxApiError("TreeDX repository ID is required.", {
131
+ status: 400,
132
+ code: "missing_repo_id"
133
+ });
134
+ }
135
+ return repoId;
136
+ }
137
+ query(params = {}) {
138
+ const search = new URLSearchParams();
139
+ for (const [key, value] of Object.entries(params)) {
140
+ if (value === void 0 || value === null) {
141
+ continue;
142
+ }
143
+ search.set(key, String(value));
144
+ }
145
+ const rendered = search.toString();
146
+ return rendered ? `?${rendered}` : "";
147
+ }
148
+ headers(bodyPresent) {
149
+ const headers = {
150
+ accept: "application/json"
151
+ };
152
+ if (bodyPresent) {
153
+ headers["content-type"] = "application/json";
154
+ }
155
+ if (this.token) {
156
+ headers.authorization = `Bearer ${this.token}`;
157
+ }
158
+ return headers;
159
+ }
160
+ async fetchWithTimeout(input, init) {
161
+ const timeoutMs = this.options.timeoutMs;
162
+ let timeout;
163
+ let controller;
164
+ if (timeoutMs && timeoutMs > 0) {
165
+ controller = new AbortController();
166
+ timeout = setTimeout(() => controller?.abort(), timeoutMs);
167
+ }
168
+ try {
169
+ return await this.fetchImpl(input, {
170
+ ...init,
171
+ signal: controller?.signal ?? init.signal
172
+ });
173
+ } catch (error) {
174
+ if (isAbortError(error)) {
175
+ throw new TreeDxApiError(`TreeDX request timed out after ${timeoutMs}ms.`, {
176
+ status: 0,
177
+ code: "timeout",
178
+ details: { timeoutMs },
179
+ payload: error
180
+ });
181
+ }
182
+ throw new TreeDxApiError(error instanceof Error ? error.message : "TreeDX network request failed.", {
183
+ status: 0,
184
+ code: "network_error",
185
+ payload: error
186
+ });
187
+ } finally {
188
+ if (timeout) {
189
+ clearTimeout(timeout);
190
+ }
191
+ }
192
+ }
193
+ async request(method, path, body, options = {}) {
194
+ if (options.tokenRequired && !this.token) {
195
+ throw new TreeDxApiError("TreeDX bearer token is required.", {
196
+ status: 401,
197
+ code: "missing_token"
198
+ });
199
+ }
200
+ const response = await this.fetchWithTimeout(`${this.baseUrl}${path}${this.query(options.query ?? {})}`, {
201
+ method,
202
+ headers: this.headers(body !== void 0),
203
+ body: body === void 0 ? void 0 : JSON.stringify(body)
204
+ });
205
+ let payload;
206
+ try {
207
+ payload = await response.json();
208
+ } catch (error) {
209
+ throw new TreeDxApiError("TreeDX response was not valid JSON.", {
210
+ status: response.status,
211
+ code: "invalid_response",
212
+ payload: error
213
+ });
214
+ }
215
+ if (!response.ok || isRecord(payload) && payload.ok === false) {
216
+ const errorBody = isRecord(payload) && isRecord(payload.error) ? payload.error : {};
217
+ throw new TreeDxApiError(
218
+ typeof errorBody.message === "string" ? errorBody.message : `TreeDX request failed with status ${response.status}.`,
219
+ {
220
+ status: response.status,
221
+ code: typeof errorBody.code === "string" ? errorBody.code : "treedx_api_error",
222
+ details: isRecord(errorBody.details) ? errorBody.details : {},
223
+ payload
224
+ }
225
+ );
226
+ }
227
+ return stripOk(payload);
228
+ }
229
+ async requestBinary(path, body, options = {}) {
230
+ if (options.tokenRequired && !this.token) {
231
+ throw new TreeDxApiError("TreeDX bearer token is required.", {
232
+ status: 401,
233
+ code: "missing_token"
234
+ });
235
+ }
236
+ const response = await this.fetchWithTimeout(`${this.baseUrl}${path}${this.query({ ...options.query, download: true })}`, {
237
+ method: "POST",
238
+ headers: this.headers(true),
239
+ body: JSON.stringify(body)
240
+ });
241
+ if (!response.ok) {
242
+ let payload;
243
+ try {
244
+ payload = await response.json();
245
+ } catch {
246
+ payload = void 0;
247
+ }
248
+ const errorBody = isRecord(payload) && isRecord(payload.error) ? payload.error : {};
249
+ throw new TreeDxApiError(
250
+ typeof errorBody.message === "string" ? errorBody.message : `TreeDX request failed with status ${response.status}.`,
251
+ {
252
+ status: response.status,
253
+ code: typeof errorBody.code === "string" ? errorBody.code : "treedx_api_error",
254
+ details: isRecord(errorBody.details) ? errorBody.details : {},
255
+ payload
256
+ }
257
+ );
258
+ }
259
+ const content = await response.arrayBuffer();
260
+ const contentDisposition = response.headers.get("content-disposition");
261
+ return {
262
+ content,
263
+ contentType: response.headers.get("content-type"),
264
+ filename: parseFilename(contentDisposition),
265
+ checksum: response.headers.get("x-treedx-artifact-checksum") ?? void 0,
266
+ snapshotId: response.headers.get("x-treedx-snapshot-id") ?? void 0
267
+ };
268
+ }
269
+ async requestBlobDownload(input) {
270
+ if (!this.token) {
271
+ throw new TreeDxApiError("TreeDX bearer token is required.", {
272
+ status: 401,
273
+ code: "missing_token"
274
+ });
275
+ }
276
+ const response = await this.fetchWithTimeout(
277
+ `${this.baseUrl}/api/v1/workspaces/${encodeURIComponent(input.workspaceId)}/blobs/download${this.query({
278
+ path: input.path,
279
+ allowProtected: input.allowProtected
280
+ })}`,
281
+ {
282
+ method: "GET",
283
+ headers: this.headers(false)
284
+ }
285
+ );
286
+ await this.assertBinaryOk(response);
287
+ return {
288
+ content: await response.arrayBuffer(),
289
+ contentType: response.headers.get("content-type"),
290
+ contentHash: response.headers.get("x-treedx-content-hash") ?? void 0,
291
+ objectId: response.headers.get("x-treedx-object-id") ?? void 0,
292
+ source: response.headers.get("x-treedx-source")
293
+ };
294
+ }
295
+ async requestBlobUpload(input) {
296
+ if (!this.token) {
297
+ throw new TreeDxApiError("TreeDX bearer token is required.", {
298
+ status: 401,
299
+ code: "missing_token"
300
+ });
301
+ }
302
+ const headers = {
303
+ accept: "application/json",
304
+ "content-type": input.contentType ?? "application/octet-stream"
305
+ };
306
+ if (this.token) {
307
+ headers.authorization = `Bearer ${this.token}`;
308
+ }
309
+ if (input.expectedSha) {
310
+ headers["x-treedx-expected-sha"] = input.expectedSha;
311
+ }
312
+ if (input.expectedContentHash) {
313
+ headers["x-treedx-expected-content-hash"] = input.expectedContentHash;
314
+ }
315
+ if (input.allowProtected !== void 0) {
316
+ headers["x-treedx-allow-protected"] = String(input.allowProtected);
317
+ }
318
+ const response = await this.fetchWithTimeout(
319
+ `${this.baseUrl}/api/v1/workspaces/${encodeURIComponent(input.workspaceId)}/blobs/upload${this.query({
320
+ path: input.path
321
+ })}`,
322
+ {
323
+ method: "PUT",
324
+ headers,
325
+ body: input.content
326
+ }
327
+ );
328
+ const payload = await this.parseJsonResponse(response);
329
+ if (!response.ok || isRecord(payload) && payload.ok === false) {
330
+ this.throwApiError(response, payload);
331
+ }
332
+ return firstPayload(payload, ["result"]);
333
+ }
334
+ async assertBinaryOk(response) {
335
+ if (response.ok) {
336
+ return;
337
+ }
338
+ let payload;
339
+ try {
340
+ payload = await response.json();
341
+ } catch {
342
+ payload = void 0;
343
+ }
344
+ this.throwApiError(response, payload);
345
+ }
346
+ async parseJsonResponse(response) {
347
+ try {
348
+ return await response.json();
349
+ } catch (error) {
350
+ throw new TreeDxApiError("TreeDX response was not valid JSON.", {
351
+ status: response.status,
352
+ code: "invalid_response",
353
+ payload: error
354
+ });
355
+ }
356
+ }
357
+ throwApiError(response, payload) {
358
+ const errorBody = isRecord(payload) && isRecord(payload.error) ? payload.error : {};
359
+ throw new TreeDxApiError(
360
+ typeof errorBody.message === "string" ? errorBody.message : `TreeDX request failed with status ${response.status}.`,
361
+ {
362
+ status: response.status,
363
+ code: typeof errorBody.code === "string" ? errorBody.code : "treedx_api_error",
364
+ details: isRecord(errorBody.details) ? errorBody.details : {},
365
+ payload
366
+ }
367
+ );
368
+ }
369
+ health() {
370
+ return this.request("GET", "/api/v1/health");
371
+ }
372
+ ready() {
373
+ return this.request("GET", "/api/v1/ready").then((payload) => firstPayload(payload, ["readiness"]));
374
+ }
375
+ deepHealth(input = {}) {
376
+ return this.request(
377
+ "GET",
378
+ input.admin ? "/api/v1/admin/health/deep" : "/api/v1/health/deep",
379
+ void 0,
380
+ { tokenRequired: input.admin === true }
381
+ ).then((payload) => firstPayload(payload, ["health"]));
382
+ }
383
+ metrics() {
384
+ return this.request("GET", "/api/v1/metrics").then((payload) => firstPayload(payload, ["metrics"]));
385
+ }
386
+ async prometheusMetrics() {
387
+ const response = await this.fetchWithTimeout(`${this.baseUrl}/metrics`, {
388
+ method: "GET",
389
+ headers: { accept: "text/plain" }
390
+ });
391
+ if (!response.ok) {
392
+ let payload;
393
+ try {
394
+ payload = await response.json();
395
+ } catch {
396
+ payload = void 0;
397
+ }
398
+ this.throwApiError(response, payload);
399
+ }
400
+ return response.text();
401
+ }
402
+ whoami() {
403
+ return this.request("GET", "/api/v1/auth/whoami");
404
+ }
405
+ authMode() {
406
+ return this.request("GET", "/api/v1/auth/mode");
407
+ }
408
+ effectiveScope(input = {}) {
409
+ return this.request("GET", "/api/v1/policy/effective-scope", void 0, {
410
+ query: { repoId: input.repoId },
411
+ tokenRequired: true
412
+ });
413
+ }
414
+ listCapabilities() {
415
+ return this.request("GET", "/api/v1/policy/capabilities", void 0, { tokenRequired: true });
416
+ }
417
+ listCapabilityGrants(input = {}) {
418
+ return this.request("GET", "/api/v1/policy/grants", void 0, {
419
+ query: input,
420
+ tokenRequired: true
421
+ }).then((payload) => firstPayload(payload, ["grants"]));
422
+ }
423
+ putCapabilityGrant(input) {
424
+ return this.request("POST", "/api/v1/policy/grants", input, { tokenRequired: true }).then((payload) => firstPayload(payload, ["grant"]));
425
+ }
426
+ listAuditEvents(input = {}) {
427
+ return this.request(
428
+ "GET",
429
+ "/api/v1/audit/events",
430
+ void 0,
431
+ { query: input, tokenRequired: true }
432
+ );
433
+ }
434
+ planFederatedQuery(input) {
435
+ return this.request("POST", "/api/v1/federation/query/plan", input, { tokenRequired: true });
436
+ }
437
+ buildSnapshot(input = {}) {
438
+ const { repoId, ...body } = input;
439
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/snapshots/build`, body, { tokenRequired: true }).then((payload) => firstPayload(payload, ["snapshot"]));
440
+ }
441
+ getSnapshot(input) {
442
+ return this.request(
443
+ "GET",
444
+ `/api/v1/repos/${encodeURIComponent(this.repoId(input.repoId))}/snapshots/${encodeURIComponent(input.snapshotId)}`,
445
+ void 0,
446
+ { tokenRequired: true }
447
+ ).then((payload) => firstPayload(payload, ["snapshot"]));
448
+ }
449
+ exportArtifact(input = {}) {
450
+ const { repoId, ...body } = input;
451
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/artifacts/export`, body, { tokenRequired: true }).then((payload) => firstPayload(payload, ["artifact"]));
452
+ }
453
+ downloadArtifact(input = {}) {
454
+ const { repoId, ...body } = input;
455
+ return this.requestBinary(`/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/artifacts/export`, body, {
456
+ tokenRequired: true
457
+ });
458
+ }
459
+ syncMirror(input) {
460
+ const { repoId, mirrorId, ...body } = input;
461
+ return this.request(
462
+ "POST",
463
+ `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/mirrors/${encodeURIComponent(mirrorId)}/sync`,
464
+ body,
465
+ { tokenRequired: true }
466
+ );
467
+ }
468
+ fetchRemote(input) {
469
+ const { repoId, ...body } = input;
470
+ return this.request(
471
+ "POST",
472
+ `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/sync`,
473
+ body,
474
+ { tokenRequired: true }
475
+ );
476
+ }
477
+ push(input) {
478
+ const { repoId, ...body } = input;
479
+ return this.request(
480
+ "POST",
481
+ `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/push`,
482
+ body,
483
+ { tokenRequired: true }
484
+ ).then((payload) => firstPayload(payload, ["push"]));
485
+ }
486
+ checkMirrorHealth(input) {
487
+ const { repoId, mirrorId } = input;
488
+ return this.request(
489
+ "POST",
490
+ `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/mirrors/${encodeURIComponent(mirrorId)}/health`,
491
+ {},
492
+ { tokenRequired: true }
493
+ );
494
+ }
495
+ promoteMirror(input) {
496
+ const { repoId, mirrorId, ...body } = input;
497
+ return this.request(
498
+ "POST",
499
+ `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/mirrors/${encodeURIComponent(mirrorId)}/promote`,
500
+ body,
501
+ { tokenRequired: true }
502
+ );
503
+ }
504
+ compactStorage(input = {}) {
505
+ return this.request("POST", "/api/v1/admin/storage/compact", input, {
506
+ tokenRequired: true
507
+ });
508
+ }
509
+ backupStorage(input = {}) {
510
+ return this.request("POST", "/api/v1/admin/storage/backup", input, {
511
+ tokenRequired: true
512
+ });
513
+ }
514
+ listStorageMigrations() {
515
+ return this.request(
516
+ "GET",
517
+ "/api/v1/admin/storage/migrations",
518
+ void 0,
519
+ { tokenRequired: true }
520
+ );
521
+ }
522
+ planStorageMigration(input = {}) {
523
+ return this.request(
524
+ "POST",
525
+ "/api/v1/admin/storage/migrations/plan",
526
+ input,
527
+ { tokenRequired: true }
528
+ ).then((payload) => firstPayload(payload, ["migration"]));
529
+ }
530
+ applyStorageMigration(input = {}) {
531
+ return this.request(
532
+ "POST",
533
+ "/api/v1/admin/storage/migrations/apply",
534
+ input,
535
+ { tokenRequired: true }
536
+ ).then((payload) => firstPayload(payload, ["migration"]));
537
+ }
538
+ rollbackStorageMigration(input) {
539
+ return this.request(
540
+ "POST",
541
+ "/api/v1/admin/storage/migrations/rollback",
542
+ input,
543
+ { tokenRequired: true }
544
+ ).then((payload) => firstPayload(payload, ["migration"]));
545
+ }
546
+ verifyStorageRestore(input) {
547
+ return this.request(
548
+ "POST",
549
+ "/api/v1/admin/storage/restore/verify",
550
+ input,
551
+ { tokenRequired: true }
552
+ ).then((payload) => payload.restore);
553
+ }
554
+ restoreStorage(input) {
555
+ return this.request(
556
+ "POST",
557
+ "/api/v1/admin/storage/restore",
558
+ input,
559
+ { tokenRequired: true }
560
+ ).then((payload) => payload.restore);
561
+ }
562
+ createMigration(input) {
563
+ const { repoId, ...body } = input;
564
+ return this.request(
565
+ "POST",
566
+ `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/migrations`,
567
+ body,
568
+ { tokenRequired: true }
569
+ );
570
+ }
571
+ getMigration(input) {
572
+ return this.request(
573
+ "GET",
574
+ `/api/v1/repos/${encodeURIComponent(this.repoId(input.repoId))}/migrations/${encodeURIComponent(input.migrationId)}`,
575
+ void 0,
576
+ { tokenRequired: true }
577
+ ).then((payload) => firstPayload(payload, ["migration"]));
578
+ }
579
+ getNode() {
580
+ return this.request("GET", "/api/v1/node", void 0, { tokenRequired: true }).then((payload) => firstPayload(payload, ["node"]));
581
+ }
582
+ listNodes() {
583
+ return this.request("GET", "/api/v1/registry/nodes", void 0, { tokenRequired: true }).then((payload) => firstPayload(payload, ["nodes"]));
584
+ }
585
+ getPlacement(repoId) {
586
+ return this.request("GET", `/api/v1/registry/repos/${encodeURIComponent(repoId)}/placement`, void 0, { tokenRequired: true }).then((payload) => firstPayload(payload, ["placement"]));
587
+ }
588
+ getRepository(repoId) {
589
+ return this.request("GET", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}`, void 0, { tokenRequired: true }).then((payload) => firstPayload(payload, ["repo"]));
590
+ }
591
+ createWorkspace(input) {
592
+ const { repoId, ...body } = input;
593
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/workspaces`, body, { tokenRequired: true });
594
+ }
595
+ async closeWorkspace(workspaceId) {
596
+ await this.request("POST", `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/close`, {}, { tokenRequired: true });
597
+ }
598
+ listTree(input) {
599
+ return this.request("GET", `/api/v1/workspaces/${encodeURIComponent(input.workspaceId)}/tree`, void 0, {
600
+ query: { path: input.path ?? "", includeDeleted: input.includeDeleted },
601
+ tokenRequired: true
602
+ }).then((payload) => firstPayload(payload, ["entries"]));
603
+ }
604
+ readFile(input) {
605
+ return this.request("GET", `/api/v1/workspaces/${encodeURIComponent(input.workspaceId)}/files`, void 0, {
606
+ query: { path: input.path, allowProtected: input.allowProtected },
607
+ tokenRequired: true
608
+ });
609
+ }
610
+ writeFile(input) {
611
+ const { workspaceId, path, ...body } = input;
612
+ return this.request("PUT", `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/files`, body, {
613
+ query: { path },
614
+ tokenRequired: true
615
+ });
616
+ }
617
+ patchFile(input) {
618
+ const { workspaceId, path, ...body } = input;
619
+ return this.request("PATCH", `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/files`, body, {
620
+ query: { path },
621
+ tokenRequired: true
622
+ });
623
+ }
624
+ deleteFile(input) {
625
+ const { workspaceId, path, ...body } = input;
626
+ return this.request("DELETE", `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/files`, body, {
627
+ query: { path },
628
+ tokenRequired: true
629
+ });
630
+ }
631
+ readBlob(input) {
632
+ const { repoId, ...body } = input;
633
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/blobs/read`, body, {
634
+ tokenRequired: true
635
+ }).then((payload) => firstPayload(payload, ["blob"]));
636
+ }
637
+ writeBlob(input) {
638
+ const { workspaceId, path, ...body } = input;
639
+ return this.request(
640
+ "POST",
641
+ `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/blobs/write`,
642
+ { path, ...body, encoding: body.encoding ?? "base64" },
643
+ { tokenRequired: true }
644
+ ).then((payload) => firstPayload(payload, ["result"]));
645
+ }
646
+ deleteBlob(input) {
647
+ const { workspaceId, ...body } = input;
648
+ return this.request(
649
+ "POST",
650
+ `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/blobs/delete`,
651
+ body,
652
+ { tokenRequired: true }
653
+ ).then((payload) => firstPayload(payload, ["result"]));
654
+ }
655
+ downloadBlob(input) {
656
+ return this.requestBlobDownload(input);
657
+ }
658
+ uploadBlob(input) {
659
+ return this.requestBlobUpload(input);
660
+ }
661
+ createBlobUpload(input) {
662
+ const { workspaceId, ...body } = input;
663
+ return this.request(
664
+ "POST",
665
+ `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/blobs/uploads`,
666
+ body,
667
+ { tokenRequired: true }
668
+ ).then((payload) => firstPayload(payload, ["upload"]));
669
+ }
670
+ async uploadBlobPart(input) {
671
+ if (!this.token) {
672
+ throw new TreeDxApiError("TreeDX bearer token is required.", {
673
+ status: 401,
674
+ code: "missing_token"
675
+ });
676
+ }
677
+ const response = await this.fetchWithTimeout(
678
+ `${this.baseUrl}/api/v1/workspaces/${encodeURIComponent(input.workspaceId)}/blobs/uploads/${encodeURIComponent(input.uploadId)}/parts/${encodeURIComponent(input.partNumber)}`,
679
+ {
680
+ method: "PUT",
681
+ headers: {
682
+ accept: "application/json",
683
+ authorization: `Bearer ${this.token}`,
684
+ "content-type": "application/octet-stream"
685
+ },
686
+ body: input.content
687
+ }
688
+ );
689
+ const payload = await this.parseJsonResponse(response);
690
+ if (!response.ok || isRecord(payload) && payload.ok === false) {
691
+ this.throwApiError(response, payload);
692
+ }
693
+ return firstPayload(payload, ["part"]);
694
+ }
695
+ completeBlobUpload(input) {
696
+ const { workspaceId, uploadId, ...body } = input;
697
+ return this.request(
698
+ "POST",
699
+ `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/blobs/uploads/${encodeURIComponent(uploadId)}/complete`,
700
+ body,
701
+ { tokenRequired: true }
702
+ ).then((payload) => firstPayload(payload, ["result"]));
703
+ }
704
+ abortBlobUpload(input) {
705
+ return this.request(
706
+ "DELETE",
707
+ `/api/v1/workspaces/${encodeURIComponent(input.workspaceId)}/blobs/uploads/${encodeURIComponent(input.uploadId)}`,
708
+ void 0,
709
+ { tokenRequired: true }
710
+ ).then((payload) => firstPayload(payload, ["upload"]));
711
+ }
712
+ listArtifacts(input = {}) {
713
+ return this.request(
714
+ "GET",
715
+ `/api/v1/repos/${encodeURIComponent(this.repoId(input.repoId))}/artifacts`,
716
+ void 0,
717
+ { tokenRequired: true }
718
+ ).then((payload) => firstPayload(payload, ["artifacts"]));
719
+ }
720
+ getArtifact(input) {
721
+ return this.request(
722
+ "GET",
723
+ `/api/v1/repos/${encodeURIComponent(this.repoId(input.repoId))}/artifacts/${encodeURIComponent(input.artifactId)}`,
724
+ void 0,
725
+ { tokenRequired: true }
726
+ ).then((payload) => firstPayload(payload, ["artifact"]));
727
+ }
728
+ deleteArtifact(input) {
729
+ return this.request(
730
+ "DELETE",
731
+ `/api/v1/repos/${encodeURIComponent(this.repoId(input.repoId))}/artifacts/${encodeURIComponent(input.artifactId)}`,
732
+ void 0,
733
+ { tokenRequired: true }
734
+ ).then((payload) => firstPayload(payload, ["artifact"]));
735
+ }
736
+ cleanupArtifacts(input = {}) {
737
+ return this.request(
738
+ "POST",
739
+ "/api/v1/admin/artifacts/cleanup",
740
+ input,
741
+ { tokenRequired: true }
742
+ ).then((payload) => firstPayload(payload, ["cleanup"]));
743
+ }
744
+ search(input) {
745
+ const { workspaceId, ...body } = input;
746
+ return this.request("POST", `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/search`, body, { tokenRequired: true });
747
+ }
748
+ status(input) {
749
+ return this.request("GET", `/api/v1/workspaces/${encodeURIComponent(input.workspaceId)}/status`, void 0, { tokenRequired: true });
750
+ }
751
+ diff(input) {
752
+ return this.request("GET", `/api/v1/workspaces/${encodeURIComponent(input.workspaceId)}/diff`, void 0, { tokenRequired: true });
753
+ }
754
+ commit(input) {
755
+ const { workspaceId, ...body } = input;
756
+ return this.request("POST", `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/commit`, body, { tokenRequired: true });
757
+ }
758
+ exec(input) {
759
+ const { workspaceId, ...body } = input;
760
+ return this.request("POST", `/api/v1/workspaces/${encodeURIComponent(workspaceId)}/exec`, body, { tokenRequired: true });
761
+ }
762
+ readRepositoryFiles(input) {
763
+ const { repoId, ...body } = input;
764
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/files/read`, body, { tokenRequired: true });
765
+ }
766
+ readRepositoryFile(input) {
767
+ return this.readRepositoryFiles(input);
768
+ }
769
+ listRepositoryPaths(input) {
770
+ const { repoId, ...body } = input;
771
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/paths/list`, body, { tokenRequired: true });
772
+ }
773
+ searchRepositoryFiles(input) {
774
+ const { repoId, ...body } = input;
775
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/files/search`, body, { tokenRequired: true });
776
+ }
777
+ queryRepository(input) {
778
+ const { repoId, ...body } = input;
779
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/query`, body, { tokenRequired: true });
780
+ }
781
+ federatedSearch(input) {
782
+ return this.request("POST", "/api/v1/search", input, { tokenRequired: true }).then((payload) => firstPayload(payload, ["search"]));
783
+ }
784
+ federatedQuery(input) {
785
+ return this.request("POST", "/api/v1/query", input, { tokenRequired: true }).then((payload) => firstPayload(payload, ["query"]));
786
+ }
787
+ federatedContext(input) {
788
+ return this.request("POST", "/api/v1/context/build", input, { tokenRequired: true }).then((payload) => firstPayload(payload, ["context"]));
789
+ }
790
+ federatedGraph(input) {
791
+ return this.request("POST", "/api/v1/graph/query", input, { tokenRequired: true }).then((payload) => firstPayload(payload, ["graph"]));
792
+ }
793
+ refreshGraph(input = {}) {
794
+ const { repoId, ...body } = input;
795
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/graph/refresh`, body, { tokenRequired: true });
796
+ }
797
+ getGraphRefreshJob(input) {
798
+ return this.request(
799
+ "GET",
800
+ `/api/v1/repos/${encodeURIComponent(this.repoId(input.repoId))}/graph/refresh-jobs/${encodeURIComponent(input.jobId)}`,
801
+ void 0,
802
+ { query: { ref: input.ref }, tokenRequired: true }
803
+ ).then((payload) => firstPayload(payload, ["job"]));
804
+ }
805
+ refreshSearchIndex(input = {}) {
806
+ const { repoId, ...body } = input;
807
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/search/index/refresh`, body, { tokenRequired: true }).then((payload) => firstPayload(payload, ["index"]));
808
+ }
809
+ getSearchIndexStatus(input = {}) {
810
+ return this.request(
811
+ "GET",
812
+ `/api/v1/repos/${encodeURIComponent(this.repoId(input.repoId))}/search/index/status`,
813
+ void 0,
814
+ { query: { ref: input.ref }, tokenRequired: true }
815
+ ).then((payload) => firstPayload(payload, ["index"]));
816
+ }
817
+ compactSearchIndex(input = {}) {
818
+ const { repoId, ...body } = input;
819
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/search/index/compact`, body, { tokenRequired: true }).then((payload) => firstPayload(payload, ["compact"]));
820
+ }
821
+ queryGraph(input) {
822
+ const { repoId, ...body } = input;
823
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/graph/query`, body, { tokenRequired: true });
824
+ }
825
+ searchGraphFiles(input) {
826
+ return this.graphSearch("/graph/search-files", input);
827
+ }
828
+ searchGraphSections(input) {
829
+ return this.graphSearch("/graph/search-sections", input);
830
+ }
831
+ searchGraphEntities(input) {
832
+ return this.graphSearch("/graph/search-entities", input);
833
+ }
834
+ getGraphNode(input) {
835
+ return this.request("GET", `/api/v1/repos/${encodeURIComponent(this.repoId(input.repoId))}/graph/nodes/${encodeURIComponent(input.nodeId)}`, void 0, {
836
+ query: { ref: input.ref },
837
+ tokenRequired: true
838
+ }).then((payload) => firstPayload(payload, ["node"]));
839
+ }
840
+ getRelated(input) {
841
+ const { repoId, ...body } = input;
842
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/graph/related`, body, { tokenRequired: true });
843
+ }
844
+ getSubgraph(input) {
845
+ const { repoId, ...body } = input;
846
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/graph/subgraph`, body, { tokenRequired: true });
847
+ }
848
+ buildContext(input) {
849
+ const { repoId, ...body } = input;
850
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/context/build`, body, { tokenRequired: true });
851
+ }
852
+ parseContextDsl(input) {
853
+ const { repoId, ...body } = input;
854
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}/context/parse-ctx`, body, { tokenRequired: true });
855
+ }
856
+ graphSearch(path, input) {
857
+ const { repoId, ...body } = input;
858
+ return this.request("POST", `/api/v1/repos/${encodeURIComponent(this.repoId(repoId))}${path}`, body, { tokenRequired: true }).then((payload) => firstPayload(payload, ["results"]));
859
+ }
860
+ }
861
+ function parseFilename(contentDisposition) {
862
+ if (!contentDisposition) {
863
+ return void 0;
864
+ }
865
+ const match = /filename="([^"]+)"/u.exec(contentDisposition) ?? /filename=([^;]+)/u.exec(contentDisposition);
866
+ return match?.[1];
867
+ }
868
+ export {
869
+ TREEDX_CLIENT_OPERATION_MAP,
870
+ TreeDxClient
871
+ };